[BZOJ3295] [Cqoi2011]动态逆序对 CDQ分治

2023-12-07 06:58

本文主要是介绍[BZOJ3295] [Cqoi2011]动态逆序对 CDQ分治,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

CDQ跑的比分治快得多

首先 我们可以把每一个点看成一个三元组(x, y, z) x 表示它当前的值 y 表示的在序列中的编号 z 表示它的时间 即第z次操作后的这个点

所以 如果某个点P在平面上的左上方有点(值小于P并且位置在P之后) 后者右下方(恰好相反)的地方有点 就会形成一个逆序对

在一开始我们很容易求出每一个点形成的逆序对总数 每次删除的时候从ans中减去

然而 在CDQ分治的过程中 树状数组需要多次使用 每次清空需要耗费大量时间 这个时候我们加入一个时间轴 只有在当前时间的点才进行统计 这样就避免了清空的操作 节省了时间

还有一个问题 每次一个点x删除时 会减去他的逆序对(x, y) y删除的时候也同样会减去 这样就重复了 要怎样处理这样的问题呢?

我们将已经删除的元素看作一个删除序列 定义d[i]为删除该元素后删除序列中新增的逆序对的个数 这就是减重复了的个数 每次删除一个数过后ans要加上d[i];

这样 每一个删除操作都成为了一个三元组(x, y, id) 我们可以用CDQ根据id排序 但是还剩下两位 处理的时候会有困难 所以我们把整个序列按y排序 这样就可以了

#include<cstdio>
#include<algorithm>
#include<cstring>
#include<iostream>
#include<queue>
#include<map>
#define lowbit(x) ((x) & (-(x)))
#define SF scanf
#define PF printf
using namespace std;
typedef long long LL;
const int MAXN = 100000;
int cnt[MAXN+10], A[MAXN+10], idx[MAXN+10], c[MAXN+10], tim[MAXN+10]; // tim 节约每次删除树状数组的时间 cnt 计算某个数形成的逆序对数量
int d[MAXN+10];
int n, m, tot;
LL ans;
struct Node {int x, y, id; // y 原序列位置 x 现在的值bool operator < (const Node &t) const {return y < t.y;}
} q[MAXN+10], tmp[MAXN+10];
void add(int x, int f) { // f = -1 询问前方 f = 1 询问后方for(; x <= n && x; x += lowbit(x) * f) {if(tim[x] != tot) c[x] = 0, tim[x] = tot;c[x]++;}
}
int query(int x, int f) {int ret = 0;for( ; x && x <= n; x += lowbit(x) * f) if(tim[x] == tot)ret += c[x];return ret;
}
void CDQ(int L, int R) {if(L == R) {PF("%lld\n", ans);ans -= cnt[q[L].y];ans += d[L];return ;}int mid = (L+R) >> 1, l1, l2;l1 = L; l2 = mid+1;for(int i = L; i <= R; i++) if(q[i].id <= mid) tmp[l1++] = q[i];else tmp[l2++] = q[i];for(int i = L; i <= R; i++) q[i] = tmp[i];CDQ(L, mid);tot++;int j = L;for(int i = mid+1; i <= R; i++) {while(j <= mid && q[j].y < q[i].y) {add(q[j].x, -1); j++;}d[q[i].id] += query(q[i].x, 1);}tot++; j = mid;for(int i = R; i > mid; i--) {while(j >= L && q[j].y > q[i].y) {add(q[j].x, 1);j--;}d[q[i].id] += query(q[i].x, -1);}CDQ(mid+1, R);l1 = L; l2 = mid+1;for(int i = L; i <= R; i++) if(l1 <= mid && (l2 > R || q[l1] < q[l2])) tmp[i] = q[l1++];else tmp[i] = q[l2++];for(int i = L; i <= R; i++) q[i] = tmp[i];
}
int main() {SF("%d%d", &n, &m);for(int i = 1; i <= n; i++) {SF("%d", &A[i]); idx[A[i]] = i;}for(int i = 1; i <= n; i++) {cnt[i] = query(A[i], 1);add(A[i], -1);ans += cnt[i];}tot++; // 时间轴移动 防止重复统计for(int i = n; i; i--) {cnt[i] += query(A[i], -1);add(A[i], 1);}for(int i = 1; i <= m; i++) {SF("%d", &q[i].x); q[i].y = idx[q[i].x];q[i].id = i;}sort(q+1, q+1+m);CDQ(1, m);
}


这篇关于[BZOJ3295] [Cqoi2011]动态逆序对 CDQ分治的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/464912

相关文章

VUE动态绑定class类的三种常用方式及适用场景详解

《VUE动态绑定class类的三种常用方式及适用场景详解》文章介绍了在实际开发中动态绑定class的三种常见情况及其解决方案,包括根据不同的返回值渲染不同的class样式、给模块添加基础样式以及根据设... 目录前言1.动态选择class样式(对象添加:情景一)2.动态添加一个class样式(字符串添加:情

SpringCloud配置动态更新原理解析

《SpringCloud配置动态更新原理解析》在微服务架构的浩瀚星海中,服务配置的动态更新如同魔法一般,能够让应用在不重启的情况下,实时响应配置的变更,SpringCloud作为微服务架构中的佼佼者,... 目录一、SpringBoot、Cloud配置的读取二、SpringCloud配置动态刷新三、更新@R

如何用Python绘制简易动态圣诞树

《如何用Python绘制简易动态圣诞树》这篇文章主要给大家介绍了关于如何用Python绘制简易动态圣诞树,文中讲解了如何通过编写代码来实现特定的效果,包括代码的编写技巧和效果的展示,需要的朋友可以参考... 目录代码:效果:总结 代码:import randomimport timefrom math

Java中JSON字符串反序列化(动态泛型)

《Java中JSON字符串反序列化(动态泛型)》文章讨论了在定时任务中使用反射调用目标对象时处理动态参数的问题,通过将方法参数存储为JSON字符串并进行反序列化,可以实现动态调用,然而,这种方式容易导... 需求:定时任务扫描,反射调用目标对象,但是,方法的传参不是固定的。方案一:将方法参数存成jsON字

.NET利用C#字节流动态操作Excel文件

《.NET利用C#字节流动态操作Excel文件》在.NET开发中,通过字节流动态操作Excel文件提供了一种高效且灵活的方式处理数据,本文将演示如何在.NET平台使用C#通过字节流创建,读取,编辑及保... 目录用C#创建并保存Excel工作簿为字节流用C#通过字节流直接读取Excel文件数据用C#通过字节

第10章 中断和动态时钟显示

第10章 中断和动态时钟显示 从本章开始,按照书籍的划分,第10章开始就进入保护模式(Protected Mode)部分了,感觉从这里开始难度突然就增加了。 书中介绍了为什么有中断(Interrupt)的设计,中断的几种方式:外部硬件中断、内部中断和软中断。通过中断做了一个会走的时钟和屏幕上输入字符的程序。 我自己理解中断的一些作用: 为了更好的利用处理器的性能。协同快速和慢速设备一起工作

动态规划---打家劫舍

题目: 你是一个专业的小偷,计划偷窃沿街的房屋。每间房内都藏有一定的现金,影响你偷窃的唯一制约因素就是相邻的房屋装有相互连通的防盗系统,如果两间相邻的房屋在同一晚上被小偷闯入,系统会自动报警。 给定一个代表每个房屋存放金额的非负整数数组,计算你 不触动警报装置的情况下 ,一夜之内能够偷窃到的最高金额。 思路: 动态规划五部曲: 1.确定dp数组及含义 dp数组是一维数组,dp[i]代表

Codeforces Round #240 (Div. 2) E分治算法探究1

Codeforces Round #240 (Div. 2) E  http://codeforces.com/contest/415/problem/E 2^n个数,每次操作将其分成2^q份,对于每一份内部的数进行翻转(逆序),每次操作完后输出操作后新序列的逆序对数。 图一:  划分子问题。 图二: 分而治之,=>  合并 。 图三: 回溯:

代码随想录冲冲冲 Day39 动态规划Part7

198. 打家劫舍 dp数组的意义是在第i位的时候偷的最大钱数是多少 如果nums的size为0 总价值当然就是0 如果nums的size为1 总价值是nums[0] 遍历顺序就是从小到大遍历 之后是递推公式 对于dp[i]的最大价值来说有两种可能 1.偷第i个 那么最大价值就是dp[i-2]+nums[i] 2.不偷第i个 那么价值就是dp[i-1] 之后取这两个的最大值就是d

LeetCode:64. 最大正方形 动态规划 时间复杂度O(nm)

64. 最大正方形 题目链接 题目描述 给定一个由 0 和 1 组成的二维矩阵,找出只包含 1 的最大正方形,并返回其面积。 示例1: 输入: 1 0 1 0 01 0 1 1 11 1 1 1 11 0 0 1 0输出: 4 示例2: 输入: 0 1 1 0 01 1 1 1 11 1 1 1 11 1 1 1 1输出: 9 解题思路 这道题的思路是使用动态规划