题目描述

排排坐,吃果果,生果甜嗦嗦,大家笑呵呵。你一个,我一个,大的分给你,小的留给我,吃完果果唱支歌,大家乐和和。红星幼儿园的小朋友们排起了长长地队伍,准备吃果果。不过因为小朋友们的身高有所区别,排成的队伍高低错乱,极不美观。设第i个小朋友的身高为hi,我们定义一个序列的杂乱程度为:满足i<j且hi>hj的(i,j)数量。幼儿园阿姨每次会选出两个小朋友,交换他们的位置,请你帮忙计算出每次交换后,序列的杂乱程度。为方便幼儿园阿姨统计,在未进行任何交换操作时,你也应该输出该序列的杂乱程度。

输入

第一行为一个正整数n,表示小朋友的数量;第二行包含n个由空格分隔的正整数h1,h2,…,hn,依次表示初始队列中小朋友的身高;第三行为一个正整数m,表示交换操作的次数;以下m行每行包含两个正整数ai和bi,表示交换位置ai与位置bi的小朋友。

输出

输出文件共m行,第i行一个正整数表示交换操作i结束后,序列的杂乱程度。

样例输入

3
130 150 140
2
2 3
1 3

样例输出

1
0
3


题解

分块+树状数组

题目描述不清,这里已补好,所求即逆序对的个数。

求逆序对我们可以使用树状数组。但是树状数组是离线的,也就是说每次交换都必须重新扫一遍,这样肯定会T。

由于每次交换时,除了这两个数及它们之间的数以外都是不需要改动的,只要找出中间的即可。

我们有分块大法。

把所有元素分成√n 个块,对每个块建立一个树状数组,就可以得出两个数之间所有整块的不同范围内数的个数。

然后对于多出来的那些数,直接暴力扫一下即可。由于它们都不是整块,所以不会有超过√n 个数。

这里偷了点懒没有if else,把符合条件加加减减直接变成加减条件成立性,应该不难理解。

时间复杂度O(n*√n*logn)。

需要注意的是两个数在同一个块内的处理,以及x>y的特判。

还有,题目中可能会出现相同的数,因此不能看作非小即大,需要分开处理。

#include <cstdio>
#include <cmath>
#include <algorithm>
using namespace std;
struct data
{
int h , p;
}a[20010];
int v[20010] , top , f[150][20010];
bool cmp1(data a , data b)
{
return a.h < b.h;
}
bool cmp2(data a , data b)
{
return a.p < b.p;
}
void update(int p , int x , int a)
{
int i;
for(i = x ; i <= top ; i += i & (-i))
f[p][i] += a;
}
int query(int p , int x)
{
int i , ans = 0;
for(i = x ; i ; i -= i & (-i))
ans += f[p][i];
return ans;
}
int main()
{
int n , m , i , j , si , ans = 0 , x , y;
scanf("%d" , &n);
si = (int)sqrt(n);
for(i = 0 ; i < n ; i ++ )
scanf("%d" , &a[i].h) , a[i].p = i;
sort(a , a + n , cmp1);
for(i = 0 ; i < n ; i ++ )
{
if(a[i].h != v[top]) v[++top] = a[i].h;
a[i].h = top;
}
sort(a , a + n , cmp2);
for(i = 0 ; i < n ; i ++ )
{
for(j = 0 ; j <= i / si ; j ++ ) ans -= query(j , a[i].h);
ans += i;
update(i / si , a[i].h , 1);
}
printf("%d\n" , ans);
scanf("%d" , &m);
while(m -- )
{
scanf("%d%d" , &x , &y);
x -- ; y -- ;
if(x > y) swap(x , y);
if(x / si == y / si)
for(i = x + 1 ; i < y ; i ++ )
ans += (a[i].h > a[x].h) + (a[i].h < a[y].h) - (a[i].h < a[x].h) - (a[i].h > a[y].h);
else
{
for(i = x / si + 1 ; i < y / si ; i ++ )
ans += (si - query(i , a[x].h)) + query(i , a[y].h - 1) - query(i , a[x].h - 1) - (si - query(i , a[y].h));
for(i = x + 1 ; i < (x / si + 1) * si ; i ++ )
ans += (a[i].h > a[x].h) + (a[i].h < a[y].h) - (a[i].h < a[x].h) - (a[i].h > a[y].h);
for(i = y / si * si ; i < y ; i ++ )
ans += (a[i].h > a[x].h) + (a[i].h < a[y].h) - (a[i].h < a[x].h) - (a[i].h > a[y].h);
}
ans += (a[x].h < a[y].h) - (a[x].h > a[y].h);
update(x / si , a[x].h , -1) , update(y / si , a[y].h , -1);
swap(a[x].h , a[y].h);
update(x / si , a[x].h , 1) , update(y / si , a[y].h , 1);
printf("%d\n" , ans);
}
return 0;
}

最新文章

  1. 【转】我应该直接学Swift还是Objective-C?
  2. caffe windows学习:第一个测试程序
  3. mysql慢查询优化之explain的各列含义
  4. ActivePython2.7 +Firefly1.2.2+WIN7服务器搭建过程(已通过)
  5. [转贴]C编译过程概述
  6. Qt中绘图坐标QPainter,Viewport与Window的关系
  7. CSS 3 属性学习大纲
  8. Web 前端开发环境
  9. Linux C/C++计划Shell命令大杂烩(1)
  10. Python3调用企业微信用于告警
  11. DBA_TABLES ALL_TABLES USER_TABLES
  12. java基础系列--volatile关键字
  13. 【原创】分布式之elk日志架构的演进
  14. python 文件下载
  15. Day6------------软连接和硬链接
  16. SaltStack 安装、简单配置和远程执行
  17. Jmeter&#160;测试计划元素详解
  18. Windows10 中在指定目录下启动Powershell
  19. tensorflow.reshap(tensor,shape,name)的使用说明
  20. VUE脚手架,babel转码 常用命令

热门文章

  1. Linux入门-第六周
  2. weex踩坑记录
  3. Delphi方法
  4. mysql日志管理#慢日志详解
  5. Python协程中使用上下文
  6. 自己动手编写 Dockerfile 构建自定义的Jenkins
  7. (数据科学学习手札13)K-medoids聚类算法原理简介&amp;Python与R的实现
  8. Android面试收集录 Android组件
  9. 第二十四篇configparser(**)
  10. Kubernetes集群(概念篇)