线段树求逆序数(离散化)POJ 2299

来源:互联网 发布:如何修改手机淘宝评价 编辑:程序博客网 时间:2024/04/29 15:53
POJ2299题意:

给出长度为n的序列,每次只能交换相邻的两个元素,问至少要交换几次才使得该序列为递增序列。例如将"91054"变成"01459",最小交换6次,如果直接暴力,会超时。


样例:(2次测试)
Sample Input

5 序列中数的个数
9
1
0
5
4
3 序列中数的个数
1
2
3
0 n=0结束

Sample Output

6
0


其中需排序的数的范围0---999 999 999;显然数组不能开这么大,序列中数的个数N不大于500000,故先要将给出的序列离散到[1,500000]。

例: int a[] = {10000000, 10, 2000, 20, 300};
那么离散化后a[] = {5, 1, 4, 2, 3},是一个一一对应关系,而且满足原来的大小关系,离散的方法如下:
Java代码 复制代码 收藏代码
  1. 1)定义一个类,用来表示序列中的一个数。
  2. class Node implements Comparable{
  3. int val;//值
  4. int no;//序号
  5. public int compareTo(Object o) {
  6. return this.val - ((Node) o).val;
  7. }
  8. }
  9. 2)然后将所给序列用下面数组p表示,排序p并离散化.
  10. int data[] = new int[n];
  11. Node[] p=new Node[n];
  12. for (int i = 0; i < n; i++) {
  13. p[i]=new Node();
  14. p[i].val=in.nextInt();
  15. p[i].no=i;
  16. }
  17. Arrays.sort(p);
  18. for(int i=0;i<n;i++){
  19. data[p[i].no]=i+1;
  20. }//以上是使其离散化

原创粉丝点击