此题和上题略有不同,但是大体差不多,不过要把题意转换过来,题目大体意思为, 输入n, 也就是n个数,这些数为0 - (n-1), 这些数刚开始给定输入的顺序, 然后求他的逆序数,然后接着把第一个移到这个数列的末尾,这时候再求出一个逆序数,直到移动一个周期,也就是移动了n次, 求他们之中的最小的一个逆序数。

大体思路:

1. 首先建立线段树,初始化每个节点的值都为0

2. 输入原序列的同时,将原序列的逆序数求出来

  其中这个求的过程为找到它导致的逆序数为多少,就是找在它之前有多少个比它大的

3. 遍历一遍, 然后将剩下的n - 1 个序列的逆序数求出来,其中这里有个公式, 可以推导出来,就是可以这么考虑,假设这是第一次将第一个数移到数组的末尾,假设这个数为X[i], 它所导致的  整个数组的逆序数的变化为: 首先,它造成他后面数的逆序数为X[i](这里的X[i]就是它本身), 这个是当前数组中把它去掉之后减少的逆序数, 所以去掉它之后整个数组的逆序数就是 Sum -   x[i](这里Sum就是总的逆序数), 然后把它加到最后,它可以增加的逆序数就为 n - 1 - x[i], 所以变化这一次当前数组的逆序数为 Sum - x[i] + n - 1 - x[i];  这个可以这么求公式,是因  为这个题特殊, 因为数组里面的数为从0 - n-1, 而且还没有重复的。

下面是代码的实现:

 #include <cstdio>
#include <algorithm> using namespace std; const int MAX = * ;
int sum[MAX];
//更新跟其有关的以上所有节点
void pushUp(int root)
{
sum[root] = sum[root * ] + sum[root * + ];
} void buildTree(int root, int left, int right)
{
sum[root] = ;//初始化
if(left == right)
{
return;
}
int mid = (left + right) / ;
buildTree(root * , left, mid);
buildTree(root * + , mid + , right);
//pushUp(root);//这里可以不需要, 因为初始化都是0
} void update(int root, int pos, int left, int right)
{
if(left == right)
{
sum[root]++;
return;
}
int mid = (left + right) / ;
if(pos <= mid)
update(root * , pos, left, mid);
else
update(root * + , pos, mid + , right);
pushUp(root);
}
//L, R代表要寻找的区间, left和right代表当前区间
int getSum(int root, int L, int R, int left, int right)
{
/*找大于L的, 因为大于L才能导致逆序数, 所以找它之前所有大于L的,
即在线段上区间可以这么写 */
if(L <= left && R >= right)
{
return sum[root];
}
int mid = (left + right) / ;
int res = ;
if(L <= mid)
{
res += getSum(root * , L, R, left, mid);
}
if(R > mid)
{
res += getSum(root * + , L, R, mid + , right);
}
return res;
} int main()
{
int x[MAX];
int n;
while(~scanf("%d", &n))
{
buildTree(, , n - );
int Sum = ;
for(int i = ; i < n; i++)
{
scanf("%d", &x[i]);
//获得当前数导致的逆序数
Sum += getSum(, x[i], n - , , n - );
update(, x[i], , n - );
}
int t = Sum;
for(int i = ; i < n; i++)
{
//此公式是由 Sum = Sum - x[i] + n - 1 - x[i]的变形
Sum += n - x[i] - x[i] - ;
t = min(Sum, t);
}
printf("%d\n", t);
} return ;
}

线段树---HDU1394Minimum Inversion Number的更多相关文章

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

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

  2. hdu1394Minimum Inversion Number(线段树,求最小逆序数)

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

  3. hdu1394Minimum Inversion Number

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

  4. HDU1394-Minimum Inversion Number

    http://acm.hdu.edu.cn/showproblem.php?pid=1394 Minimum Inversion Number Time Limit: 2000/1000 MS (Ja ...

  5. C++-HDU1394-Minimum Inversion Number[数据结构][树状数组]

    给出0~n-1的一个排列,可以整体移动,求逆序对最小值 把数字num[i]的加入,等价于树状数组的第n-num[i]位加1 因为num[i]是第 (n-1)-num[i]+1=n-num[i]大的数字 ...

  6. 线段树总结 (转载 里面有扫描线类 还有NotOnlySuccess线段树大神的地址)

    转载自:http://blog.csdn.net/shiqi_614/article/details/8228102 之前做了些线段树相关的题目,开学一段时间后,想着把它整理下,完成了大牛NotOnl ...

  7. HDU 1394 Minimum Inversion Number(最小逆序数 线段树)

    Minimum Inversion Number [题目链接]Minimum Inversion Number [题目类型]最小逆序数 线段树 &题意: 求一个数列经过n次变换得到的数列其中的 ...

  8. HDU 1394 Minimum Inversion Number(最小逆序数/暴力 线段树 树状数组 归并排序)

    题目链接: 传送门 Minimum Inversion Number Time Limit: 1000MS     Memory Limit: 32768 K Description The inve ...

  9. ACM Minimum Inversion Number 解题报告 -线段树

    C - Minimum Inversion Number Time Limit:1000MS     Memory Limit:32768KB     64bit IO Format:%I64d &a ...

随机推荐

  1. 【转】lua random()

    先来看看这两段代码: ① math.randomseed(os.time())for i=1,10 do n = math.random(10) print(n) 运行结果是: 63210754341 ...

  2. uvalive 5721 Activation (概率dp+方程)

    题目链接:http://vjudge.net/problem/viewProblem.action?id=24999 主要思想就是解方程的思想. 二维dp应该很容易想到,就是当前位置加队伍长度. dp ...

  3. PHP Countable接口

    实现该接口可以使用count()方法来获取集合的总数

  4. linux c下几种定时器实现

    1.alarm n秒后触发一次,不是循环的2.setitimer 可以发出3种信号给自己,3.timerfd 这个接口基于文件描述符,通过文件描述符类似epoll那种的可读事件进行超时通知,能够被用于 ...

  5. 安装 mysql

    1.安装mysql客户端 yum install mysql 2.安装mysql 服务器端 yum install mysql-server 3.配置 mysql字符集 /etc/my.cnf 加入 ...

  6. 【转载】如何学习javascript

    如何学习Javascript 作者: chaomao  首先要说明的是,咱现在不是高手,最多还是一个半桶水,算是入了JS的门. 谈不上经验,都是一些教训. 这个时候有人要说,“靠,你丫半桶水,凭啥教我 ...

  7. 基于jq插件开发及弹窗实例

    基于jq的插件开发是什么鬼,$.fn是什么鬼,我在实际工作中也遇到过这个问题,下面就让我们一起来看一看这些都是什么鬼. 首先我们介绍$.fn. $.fn是指jquery的命名空间,加上fn上的方法及属 ...

  8. WebService传递XML数据 C#DataSet操作XML 解析WebService返回的XML数据

    Webservice传递的数据只能是序列化的数据,典型的就是xml数据.   /// <summary>         /// 通过用户名和密码 返回下行数据         /// & ...

  9. asp.net 调用天气所遇到的问题

    由于在项目用了显示天气的功能,原有的调用方法 直接通过      <iframe name="weather_inc" src="http://i.tianqi.c ...

  10. windows下Django 部署到Apache24的配置

    1.首先下载最新版Apachehttp://httpd.apache.org/download.cgi#apache24,目前官方以不提供windows msi安装包,下载好的直接解压至C盘即可,ap ...