为什么线段树能够求逆序数?

给一个简单的序列 9 5 3 他的逆序数是3

首先要求一个逆序数有两种方式:能够从头開始往后找比当前元素小的值,也能够从后往前找比当前元素大的值,有几个逆序数就是几。

线段树就是应用从后往前找较大值得个数。(一边更新一边查)

当前个数是 n = 10

元素   9  5   3



9先增加线段树,T【9】+=1;查从T【9】到T【10】比9大的值,没有sum = 0。

5 增加线段树。T【5】 += 1。查从T【5】到T【10】比5大的值。有一个9。sum +=1;

3 增加线段树,T【3】 += 1,查从T【3】到T【10】比3大的值,有两个9和5,sum +=2;

终于sum = 3;





若元素值不能确定。那么首先讲数列离散化,在从前往后枚举。找后面比前面小的统计个数……

离散化:比如2 5 8 3 10 等价于 1 3 4 2 5,能够通过排序加小小处理解决。

比如:

元素    9 5  3 4 1     离散化后得到索引

索引    1 2 3 4 5

将元素升序排序

元素    1 3 4 5 9

索引    5 3 4 2 1





    枚举到第i个数,我们须要求出从1到i-1中有多少个比a[i]大的数,更新答案。

详细怎么做呢?

    每次枚举完一个数之后。将这个数插入到线段树里(注意:插到与其相应的位置)。

举个样例:3 2 4 1。

则线段树的变化应该为:t[3]+=1;t[2]+=1;t[4]+=1;t[1]+=1;

    

  设x=a[i],这样,在插入一个数X时,首先求一下t[x+1]~t[n]的和,这个和就是1~i-1中有多少个比a[i]大的数

 由于求一个数列的逆序数能够从两个

在你插a[i]时,前面已经有多少个比a[i]大的数

POJ 2299
#include <iostream>
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <algorithm>
#include <cmath>
using namespace std;
#define MAX INT_MAX
#define MIN INT_MIN
#define LL __int64
#define init(a) memset(a,0,sizeof(a))
#define lson l , m , rt << 1
#define rson m + 1 , r , rt << 1 | 1
const int maxn = 500010;
using namespace std; int T[maxn<<2], index[maxn], num[maxn];
int n; int cmp(const int i, const int j)
{
return num[i] < num[j];
} void update( int l, int r,int rt, int num, int add)
{
T[rt] += add;
if(r==l) return ;
int m = (l+r) >> 1;
if(num <= m) update(lson, num, add);
else update(rson, num, add);
} int query( int l, int r, int rt,int L, int R)
{
if(L<=l && r<=R) return T[rt];
int m = (l+r) >> 1;
int ans = 0;
if(L <= m) ans += query(lson, L, R);
if(R > m) ans += query(rson, L, R);
return ans;
} int main()
{
while(~scanf("%d", &n))
{
if(n==0) break;
init(index);
for(int i=1;i<=n;i++)
{
scanf("%d", &num[i]);
update(1, n,1, i, 1);//离散化
index[i] = i;
}
sort(index+1, index+n+1,cmp); //排序num。更新索引
LL sum = 0;
for(int i=1;i<=n;i++)
{
sum += (query( 1, n,1, 1, index[i])-1);
update(1, n,1, index[i], -1);
}
printf("%I64d\n", sum);
}
return 0;
}

HDU 1394 

题意:一组数 每次把最前面的元素放在最后生成新的序列。再统计逆序数,求全部逆序中最小值

有一个公式:先求出原数列的逆序数。sum

那么每次新逆序数 = sum-a[i] + (n-a[i] - 1)

#include <iostream>
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <algorithm>
#include <cmath>
using namespace std;
#define MAX INT_MAX
#define MIN INT_MIN
#define LL __int64
#define init(a) memset(a,0,sizeof(a))
#define lson l , m , rt << 1
#define rson m + 1 , r , rt << 1 | 1
const int maxn = 500010;
using namespace std; int T[maxn<<2], index[maxn], num[maxn];
int n; int cmp(const int i, const int j)
{
return num[i] < num[j];
} void update( int l, int r,int rt, int num, int add)
{
T[rt] += add;
if(r==l) return ;
int m = (l+r) >> 1;
if(num <= m) update(lson, num, add);
else update(rson, num, add);
} int query( int l, int r, int rt,int L, int R)
{
if(L<=l && r<=R) return T[rt];
int m = (l+r) >> 1;
int ans = 0;
if(L <= m) ans += query(lson, L, R);
if(R > m) ans += query(rson, L, R);
return ans;
} int main()
{
while(~scanf("%d", &n))
{
if(n==0) break;
init(index);
for(int i=1;i<=n;i++)
{
scanf("%d", &num[i]);
update(1, n,1, i, 1);
index[i] = i;
}
sort(index+1, index+n+1,cmp); LL sum = 0;
for(int i=1;i<=n;i++)
{
sum += (query( 1, n,1, 1, index[i])-1);
update(1, n,1, index[i], -1); }
// printf("%I64d\n", sum);
LL ans = sum;
for(int i = 1;i<=n;i++)
{
sum = sum - num[i] + (n-num[i]- 1);
ans = min(sum,ans);
}
cout<<ans<<endl;
}
return 0;
}

线段树求逆序数方法 HDU1394&amp;&amp;POJ2299的更多相关文章

  1. HDU_1394_Minimum Inversion Number_线段树求逆序数

    Minimum Inversion Number Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java ...

  2. HDU-1394 Minimum Inversion Number(线段树求逆序数)

    Minimum Inversion Number Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Ot ...

  3. <Sicily>Inversion Number(线段树求逆序数)

    一.题目描述 There is a permutation P with n integers from 1 to n. You have to calculate its inversion num ...

  4. hdu1394--Minimum Inversion Number(线段树求逆序数,纯为练习)

    Minimum Inversion Number Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Ot ...

  5. hdu 1394 (线段树求逆序数)

    <题目链接> 题意描述: 给你一个有0--n-1数字组成的序列,然后进行这样的操作,每次将最前面一个元素放到最后面去会得到一个序列,那么这样就形成了n个序列,那么每个序列都有一个逆序数,找 ...

  6. 【线段树求逆序数】【HDU1394】Minimum Inversion Number

    题目大意: 随机给你全排列中的一个,但不断的把第一个数丢到最后去,重复N次,形成了N个排列,问你这N个排列中逆序数最小为多少 做法: 逆序数裸的是N^2 利用线段树可以降到NlogN 具体方法是插入一 ...

  7. hdu1394 Minimum Inversion Number (线段树求逆序数&&思维)

    题目传送门 Minimum Inversion Number Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K ...

  8. HDU - 1394 Minimum Inversion Number (线段树求逆序数)

    Description The inversion number of a given number sequence a1, a2, ..., an is the number of pairs ( ...

  9. HDU - 1394 Minimum Inversion Number(线段树求逆序数---点修改)

    题意:给定一个序列,求分别将前m个数移到序列最后所得到的序列中,最小的逆序数. 分析:m范围为1~n,可得n个序列,求n个序列中最小的逆序数. 1.将序列从头到尾扫一遍,用query求每个数字之前有多 ...

随机推荐

  1. 12、第十二节课,css伪类 (转)

    一.特殊选择器 1.* 用于匹配任何的标记 2.> 用于指定父子节点关系 3.E + F 毗邻元素选择器,匹配所有紧随E元素之后的同级元素F 4.E ~ F 匹配所有E元素之后的同级元素F 5. ...

  2. MVC的Model层中的一些便签

    由于自己重新接触MVC,所以把Model层里的一些标签给记录下来,方便自己的使用. 这些是自己目前试用过的一些,在以后的工作中我会接着补充进去新的内容

  3. [Linux]三种方案在Windows系统下安装ubuntu双系统(转)

    在学习linux的过程中,ubuntu无疑是初学者的最佳选择. 下面来列举给Windows系统安装ubuntu双系统的三种方法. 一.虚拟机安装(不推荐) 使用工具:Vmware 如果不是因为迫不得已 ...

  4. YII框架中php入口文件隐藏

    Apache配置修改 主要修改下httpd文件中的两个地方 1.启用mod_rewrite.so模块,在Apache的配置文件中找到如下行,去掉前面的字符"#",保存 #LoadM ...

  5. 本机运行.net 2.0项目报错,解决方案

    本机在iis上运行.net 2.0项目时,报以下错误“请求的内容似乎是脚本,因而将无法由静态文件处理程序来处理”: 本机环境配置: 安装有以下几个.net 版本:v1.0.3705,v1.1.4322 ...

  6. EF中使用Contains方法

    第一种情况 var db=new ECEntities(); var list=new []{"8","9"}; var result=from a in db ...

  7. (转)server.urlencode有什么用?怎么用?

    这个函数主要是为了,不同浏览器的事,如果你传递的参数是中文,有的浏览器就显示出错,如果转换成编码形式,就可以正确传递!当然在IE下2个结果都是一样的,你看不出来区别,用编码来传递参数是一个好习惯,养成 ...

  8. Php 基本语法

    php基本语法 1. 四种不同的开始结束标记 只有<?php ?>.<script language="php"></script>两个总是可用 ...

  9. oracle 使用 ALTER 操作列

    使用 ALTER TABLE 语句追加, 修改, 或删除列的语法

  10. 【转】Qt之模型/视图

    [本文转自]http://blog.sina.com.cn/s/blog_a6fb6cc90101hh20.html   作者: 一去丶二三里 关于Qt中MVC的介绍与使用,助手中有一节模型/视图编程 ...