题目

https://uva.onlinejudge.org/index.php?option=com_onlinejudge&Itemid=8&page=show_problem&problem=3141

题意

一个1到n的排列,每次随机删除一个,问删除前的逆序数

思路

综合考虑,对每个数点,令value为值,pos为位置,time为出现时间(总时间-消失时间),明显是统计value1 > value2, pos1 < pos2, time1 < time2的个数

首先对其中一个轴排序,比如value,这样在归并过程中,左子树的value总是小于右子树的,可以分治。

当左右子树包含哪些数点已经确定后,可以用自下而上的归并排序使得子树上的数点按照第二维相对有序,方便用尺取法统计子树之间的逆序数。

第三维通过树状数组进行压缩,加快统计速度。

注意仅仅统计左子树对右子树的影响,就会错过右子树中的数点出现的比较晚的情况。因此需要统计右子树对左子树的影响,此时注意别把同一时间出现的重复计数。

感想

1. 注意long long!!!

2. BIT的上限要>=n!

3. 注意统计影响完成后需要清空树状数组(区间大小已经减少了所以可以浪费地使用),此时不能直接用memset清空整个数组,时间会成为O(n2),超时。

代码

时间: 0.250s

时间复杂度O(cnlogn)

#include <algorithm>
#include <cassert>
#include <cmath>
#include <cstdio>
#include <cstring>
#include <queue>
#include <tuple>
#include <cassert> using namespace std; const int MAXN = int(4e5 + ); #define LEFT_CHILD(x) ((x) << 1)
#define RIGHT_CHILD(x) (((x) << 1) + 1)
#define FATHER(x) ((x) >> 1)
#define IS_LEFT_CHILD(x) (((x) & 1) == 0)
#define IS_RIGHT_CHILD(x) (((x) & 1) == 1)
#define BROTHER(x) ((x) ^ 1)
#define LOWBIT(x) ((x) & (-x)) #define LOCAL_DEBUG struct Node{
int value, pos, time;
}nodes[MAXN], tmpNodes[MAXN]; int timeCnt[MAXN * ];
long long revNum[MAXN];
int clearStack[MAXN];
int clearLen;
int n, m;
int bitLimit; int getHigherBit(int n) {
int x = ;
while (x < n) { x <<= ; }
return x;
} void update(int id) {
while (id <= bitLimit) {
if (timeCnt[id] == ) {
clearStack[clearLen++] = id;
}
timeCnt[id]++;
id += LOWBIT(id);
}
} void clearCnt() {
while (clearLen > ) {
timeCnt[clearStack[--clearLen]] = ;
}
} int countTimesSmaller(int id) {
if (id < )return ;
int sum = ;
int tmp = ;
while (id > ) {
sum += timeCnt[id];
id -= LOWBIT(id);
}
return sum;
} void merge_by_pos(int root_ind, int internal_l, int internal_r) {
int internal_mid = (internal_l + internal_r) >> ;
for (int i = internal_l; i <= internal_r; i++) {
tmpNodes[i] = nodes[i];
}
for (int i = internal_l, j = internal_mid + , ind = internal_l; ind <= internal_r; ) {
if (i > internal_mid) {
nodes[ind++] = tmpNodes[j++];
}
else if (j > internal_r) {
nodes[ind++] = tmpNodes[i++];
}
else if (tmpNodes[i].pos < tmpNodes[j].pos) {
nodes[ind++] = tmpNodes[i++];
}
else {
nodes[ind++] = tmpNodes[j++];
}
}
}
void cal(int root_ind, int internal_l, int internal_r) {
if (internal_l == internal_r)return;
int internal_mid = (internal_l + internal_r) >> ;
if(internal_l != internal_mid)cal(LEFT_CHILD(root_ind), internal_l, internal_mid);
if (internal_mid + != internal_r)cal(RIGHT_CHILD(root_ind), internal_mid + , internal_r);
// printf("L Node: %d[%d, %d] LC: %d[%d, %d], RC: %d[%d, %d]\n", root_ind, internal_l, internal_r, LEFT_CHILD(root_ind), internal_l, internal_mid, RIGHT_CHILD(root_ind), internal_mid + 1, internal_r);
for (int i = internal_l, j = internal_mid + ; i <= internal_mid; i++) {
while (j <= internal_r && nodes[i].pos > nodes[j].pos) {
update(nodes[j].time);
j++;
}
revNum[nodes[i].time] += countTimesSmaller(nodes[i].time);
// printf("L (%d, %d, %d): +%d\n", nodes[i].value, nodes[i].pos, nodes[i].time, countTimesSmaller(nodes[i].time));
}
clearCnt(); for (int i = internal_mid, j = internal_r; j > internal_mid; j--) {
while (i >= internal_l && nodes[i].pos > nodes[j].pos) {
update(nodes[i].time);
i--;
}
revNum[nodes[j].time] += countTimesSmaller(nodes[j].time - );
// printf("R (%d, %d, %d): +%d\n", nodes[j].value, nodes[j].pos, nodes[j].time, countTimesSmaller(nodes[j].time - 1));
}
clearCnt();
merge_by_pos(root_ind, internal_l, internal_r); } int main() {
#ifdef LOCAL_DEBUG
freopen("input.txt", "r", stdin);
freopen("output2.txt", "w", stdout);
#endif // LOCAL_DEBUG
for (int ti = ; scanf("%d%d", &n, &m) == ; ti++) {
bitLimit = getHigherBit(n);
for (int i = ; i <= n; i++) {
int tmp;
scanf("%d", &tmp);
nodes[tmp].value = tmp;
nodes[tmp].pos = i;
nodes[tmp].time = ;
}
for (int i = ; i <= m + ; i++) { revNum[i] = ; }
for (int i = ; i < m; i++) {
int tmp;
scanf("%d", &tmp);
nodes[tmp].time = m - i + ;
}
cal(, , n);
long long ans = ;
for (int i = ; i <= m + ; i++) { ans += revNum[i]; }
for (int i = ; i < m; i++) {
printf("%lld\n", ans);
ans -= revNum[m - i + ];
}
}
return ;
}

UVA 11990 `Dynamic'' Inversion CDQ分治, 归并排序, 树状数组, 尺取法, 三偏序统计 难度: 2的更多相关文章

  1. [APIO2019] [LOJ 3146] 路灯 (cdq分治或树状数组套线段树)

    [APIO2019] [LOJ 3146] 路灯 (cdq分治或树状数组套线段树) 题面 略 分析 首先把一组询问(x,y)看成二维平面上的一个点,我们想办法用数据结构维护这个二维平面(注意根据题意这 ...

  2. bzoj 1176 cdq分治套树状数组

    题面: 维护一个W*W的矩阵,初始值均为S.每次操作可以增加某格子的权值,或询问某子矩阵的总权值.修改操作数M<=160000,询问数Q<=10000,W<=2000000. Inp ...

  3. bzoj 4991 [Usaco2017 Feb]Why Did the Cow Cross the Road III(cdq分治,树状数组)

    题目描述 Farmer John is continuing to ponder the issue of cows crossing the road through his farm, intro ...

  4. UVA 11990 ``Dynamic'' Inversion (序列分治)

    26天以前做过的一道题,之前的做法是分治预处理,树套树在线修改,复杂度为O(nlogn+m*logn*logn),代码量较大. 本来想学习一下cdq分治的,看到论文上的凸包.斜率就暂时放一边了,只知道 ...

  5. 【BZOJ4285】使者 cdq分治+扫描线+树状数组

    [BZOJ4285]使者 Description 公元 8192 年,人类进入星际大航海时代.在不懈的努力之下,人类占领了宇宙中的 n 个行星,并在这些行星之间修建了 n - 1 条星际航道,使得任意 ...

  6. HDU 5618 Jam's problem again(三维偏序,CDQ分治,树状数组,线段树)

    Jam's problem again Time Limit: 5000/2500 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Othe ...

  7. BZOJ 2716 [Violet 3]天使玩偶 (CDQ分治、树状数组)

    题目链接: https://www.lydsy.com/JudgeOnline/problem.php?id=2716 怎么KD树跑得都那么快啊..我写的CDQ分治被暴虐 做四遍CDQ分治,每次求一个 ...

  8. 【CJOJ2616】 【HZOI 2016】偏序 I(cdq分治,树状数组)

    传送门 CJOJ Solution 考虑这是一个四维偏序对吧. 直接cdq套在一起,然后这题有两种实现方法(树状数组的更快!) 代码实现1(cdq+cdq+cdq) /* mail: mleautom ...

  9. bzoj2253纸箱堆叠(动态规划+cdq分治套树状数组)

    Description P 工厂是一个生产纸箱的工厂.纸箱生产线在人工输入三个参数 n p a , 之后,即可自动化生产三边边长为 (a mod P,a^2 mod p,a^3 mod P) (a^4 ...

随机推荐

  1. 网页常见单位: px em pt % rem vw、vh、vmin、vmax , rem 使用

    1.网页常见单位:  px  em  pt    vw\vh   rem 1.1 px单位名称为像素,相对长度单位,像素(px)是相对于显示器屏幕分辨率而言  (最终解析单位) em单位名称为相对长度 ...

  2. openvswitch vlan下的虚拟机与物理机通信

    1,安装openvswitch ,图形界面显示等用到的安装包. yum install libvirt openvswitch python-virtinst xauth tigervnc -y 2, ...

  3. TEA加密/解密算法

    在游戏项目中,一般需要对资源或数据进行加密保护,最简单高效的加密算法就是采用位与或之类的,但是比较容易被人分析出来.TEA加密算法不但比较简单,而且有很强的抗差分分析能力,加密速度也比较快.可以根据项 ...

  4. TP3.2.3框架与已有模板做结合

    具体实现步骤: a.  复制模板文件到View指定目录 b. 复制到css.img.js静态资源文件到系统指定目录 c. 把静态资源(css.img.js)文件的路径设置为"常量" ...

  5. boke例子:用户登录

    boke例子:用户登录 1.首先创建user表,authority表(角色),user_authority,表(用户角色表) Authority实体类,需要继承:GrantedAuthority类, ...

  6. Mycat入门核心概念

      Mycat中的核心概念     Mycat中的核心概念 1.数据库中间件    Mycat 是一个开源的分布式数据库系统,但是由于真正的数据库需要存储引擎,而 Mycat 并没有 存储引擎,所以并 ...

  7. LeetCode--235--二叉树的最近公共祖先

    问题描述: 给定一个二叉搜索树, 找到该树中两个指定节点的最近公共祖先. 百度百科中最近公共祖先的定义为:“对于有根树 T 的两个结点 p.q,最近公共祖先表示为一个结点 x,满足 x 是 p.q 的 ...

  8. RMQ 解决区间查询问题

    线段树写法不管,比较灵活.这里主要讨论DP实现. 其实单纯说RMQ解决的是区间最值查询是不准确的,只要满足一个区间的信息可以从它的覆盖区间获得(即[L,R]<=[L,r],[l,R] (l< ...

  9. Amaze UI——slider的参数说明

    <script type="text/javascript"> $(function(){ $('.am-slider').flexslider({ playAfter ...

  10. spring boot(十四)shiro登录认证与权限管理

    这篇文章我们来学习如何使用Spring Boot集成Apache Shiro.安全应该是互联网公司的一道生命线,几乎任何的公司都会涉及到这方面的需求.在Java领域一般有Spring Security ...