hdu1394 分治 or 线段树
利用分治求一次逆序数,然后每次把第一个元素放到末尾,设该交换元素的值为x,设上一次求得的逆序数为y,那么此时的逆序数等于y - x + (n - x - 1),减去x是因为x作为第一个元素,其后共有x个元素小于x,移动x会导致逆序数减少x个,而加上 (n - x - 1) 是因为将x移动到末尾,其前面(n - 1)个元素中会有(n - x - 1)个元素大于x。
此题的复杂度在于求第一次逆序数O(nlgn),后面每次移动元素求更新后的逆序数时间是O(1),因此总的复杂度为(nlgn)。
AC代码:
#include<cstdio>
#include<algorithm>
#include<cstring>
using namespace std;
const int maxn = 5000 +5;
int a[maxn], c[maxn];
int solve(int l, int r){
int mid = (l + r) / 2;
if( l == r) return 0;
int ans = 0;
ans += solve(l, mid) + solve(mid + 1, r);
// Merge
int b[maxn];
int x = l, y= mid + 1;
int k = 0;
while(x <= mid && y <= r){
if(a[x] <= a[y]){
b[k++] = a[x++];
}
else {
ans += mid + 1 - x;
b[k++] = a[y++];
}
}
while(x <= mid) b[k++] = a[x++];
while(y <= r) b[k++] = a[y++];
k = 0;
for(int i = l; i <= r; ++i) a[i] = b[k++];
return ans;
}
int main(){
int n;
while(scanf("%d", &n) == 1){
for(int i = 0; i < n; ++i){
scanf("%d", &a[i]);
}
memcpy(c, a, sizeof(a));
int ans = solve(0, n-1);
int x = ans;
for(int i = 0; i < n; ++i){
ans = min(ans, x + n - 1 - 2 * c[i]);
x = x + n - 1 - 2 * c[i];
}
printf("%d\n",ans);
}
return 0;
}
线段树也能做这个题,每个区间保存的值代表[l, r]中总出现多少元素,每次加入x元素时,查找区间[x + 2, n]即可得到该元素的加入会增加多少逆序数。
贴上线段树代码:
#include<cstdio>
#define min(x,y) (x) < (y) ? x : y
const int maxn = 4 * 5000 + 5;
int a[5000 + 5];
struct node{
int L, R;
int cnt;
}t[maxn];
void Build(int l, int r, int cur){
t[cur].L = l, t[cur].R = r;
t[cur].cnt = 0;
if(l == r) return;
int mid = (l + r) / 2;
Build(l, mid, cur << 1);
Build(mid + 1, r, (cur << 1) + 1);
}
void add(int c, int cur){
int l = t[cur].L, r = t[cur].R;
t[cur].cnt++;
if(l == r) return;
int mid = (l + r) / 2;
if(c <= mid) add(c, cur << 1);
else add(c, (cur << 1) + 1);
}
int find1(int l, int r, int cur){ //search the Inversion (logn)
int l1 = t[cur].L, r1 = t[cur].R;
if(l == l1 && r == r1) return t[cur].cnt;
int mid = (l1 + r1) / 2;
if(r <= mid) return find1(l, r, cur << 1);
else if(l >= mid + 1) return find1(l, r, (cur << 1) + 1);
else return find1(l, mid, cur << 1) + find1(mid + 1, r, (cur << 1) + 1);
}
int main(){
int n;
while(scanf("%d", &n) == 1){
Build(1, n, 1);
int ans = 0;
for(int i = 0; i < n; ++i){
scanf("%d", &a[i]);
a[i]++;
add(a[i], 1);
if(a[i] + 1 <= n) ans += find1(a[i] + 1, n, 1);
}
int x = ans;
for(int i = 0; i < n; ++i){
ans = min(ans, x + n + 1 - 2 * a[i]);
x = x + n + 1 - 2 * a[i];
}
printf("%d\n",ans);
}
return 0;
}
如有不当之处欢迎指出!
hdu1394 分治 or 线段树的更多相关文章
- 【BZOJ2001】[HNOI2010]城市建设(CDQ分治,线段树分治)
[BZOJ2001][HNOI2010]城市建设(CDQ分治,线段树分治) 题面 BZOJ 洛谷 题解 好神仙啊这题.原来想做一直不会做(然而YCB神仙早就切了),今天来怒写一发. 很明显这个玩意换种 ...
- BZOJ3711 Druzyny 最大值分治、线段树
传送门 被暴力包菜了,然而还不会卡-- 有一个很暴力的DP:设\(f_i\)表示给\(1\)到\(i\)分好组最多可以分多少组,转移枚举最后一个组.接下来考虑优化这个暴力. 考虑:对于每一个位置\(i ...
- BZOJ4860 BJOI2017 树的难题 点分治、线段树合并
传送门 只会线段树……关于单调队列的解法可以去看“重建计划”一题. 看到路径长度$\in [L,R]$考虑点分治.可以知道,在当前分治中心向其他点的路径中,始边(也就是分治中心到对应子树的根的那一条边 ...
- 线段树分治总结(线段树分治,线段树,并查集,树的dfn序,二分图染色)
闲话 stO猫锟学长,满脑子神仙DS 网上有不少Dalao把线段树分治也归入CDQ分治? 还是听听YCB巨佬的介绍: 狭义:只计算左边对右边的贡献. 广义:只计算外部对内部的贡献. 看来可以理解为广义 ...
- [BJOI2017]树的难题 点分治,线段树合并
[BJOI2017]树的难题 LG传送门 点分治+线段树合并. 我不会写单调队列,所以就写了好写的线段树. 考虑对于每一个分治中心,把出边按颜色排序,这样就能把颜色相同的子树放在一起处理.用一棵动态开 ...
- 【BZOJ4317】Atm的树 动态树分治+二分+线段树
[BZOJ4317]Atm的树 Description Atm有一段时间在虐qtree的题目,于是,他满脑子都是tree,tree,tree…… 于是,一天晚上他梦到自己被关在了一个有根树中,每条路径 ...
- luoguU60884 【模板】动态点分治套线段树
题目连接:https://www.luogu.org/problemnew/show/U60884 题意:有N个点,标号为1∼N,用N−1条双向带权通道连接,保证任意两个点能互相到达. Q次询问,问从 ...
- [CF1303G] Sum of Prefix Sums - 点分治,李超线段树
给定一棵 \(n\) 个点的带点权的树,求树上的路径 \(x_1,...,x_k\) ,最大化 \(\sum_{i=1}^k ia_{x_i}\) Solution 树上路径问题可用点分治. 考虑如何 ...
- ACM学习历程—HDU5696 区间的价值(分治 && RMQ && 线段树 && 动态规划)
http://acm.hdu.edu.cn/showproblem.php?pid=5696 这是这次百度之星初赛2B的第一题,但是由于正好打省赛,于是便错过了.加上2A的时候差了一题,当时有思路,但 ...
随机推荐
- Linux指令--chmod
chmod命令用于改变linux系统文件或目录的访问权限.用它控制文件或目录的访问权限.该命令有两种用法.一种是包含字母和操作符表达式的文字设定法:另一种是包含数字的数字设定法. Linux系统中的每 ...
- JavaScript算法实现排序
<!DOCTYPE html> <html> <head> <meta charset="utf-8"> <title> ...
- 2018Pycharm激活方法
1.将"0.0.0.0 account.jetbrains.com"添加到hosts文件中 2.打开http://idea.lanyus.com/ 3.获取激活码,粘贴到第二个选项 ...
- linux上的组管理
上一次我们谈了CentOS上的用户管理,现在我们再来谈下CentOS上的用户组管理. groupadd创建一个新的组 用法如下: groupadd [选项] groupname 常用选项: -f 强制 ...
- 不需要客户端插件PHP也能实现单点登录
分析CAS原理,构建PHP单点登录 单点登录(Single Sign On , 简称 SSO )是目前比较流行的服务于企业业务整合的解决方案之一, SSO 使得在多个应用系统中,用户 只需要登录一次就 ...
- 让windows系统的DOS窗口也可以显示utf8字符集
C:\Users\Administrator>chcp活动代码页: 936 windows cmd命令显示UTF8设置 在中文Windows系统中,如果一个文本文件是UTF-8编码的,那么在CM ...
- IntelliJ IDEA(七) :Project Structure
Project Structure “ 项目结构”对话框允许您管理项目和IDE级别的元素,例如Modules,Facets,Libraries, Artifacts和SDK. 在大多数情况下,左边部分 ...
- IDEA Tomcat:Failed to initialize end point associated with ProtocolHandler
发现Tomcat的日志中出现这样的错误,一般都是端口被占用了.在任务管理器中检查是否有其他的应用在使用该端口 Failed to initialize end point associated wit ...
- java中队列Queue的使用
1.在java5中新增加了java.util.Queue接口,用以支持队列的常见操作.Queue接口与List.Set同一级别,都是继承了Collection接口.Queue使用时要尽量避免Colle ...
- bzoj 3451: Tyvj1953 Normal [fft 点分治 期望]
3451: Tyvj1953 Normal 题意: N 个点的树,点分治时等概率地随机选点,代价为当前连通块的顶点数量,求代价的期望值 百年难遇的点分治一遍AC!!! 今天又去翻了一下<具体数学 ...