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的时候差了一题,当时有思路,但 ...
随机推荐
- Python起步
最近研究系统自动化测试想起了一年前有学习Python的想法,借此机会准备抽时间好好学学.为方便以后学习和查询特写以下博客! Python基础 1. Python数据结构 (1)Python字符串 (2 ...
- Servlet--Servlet接口
servlet主要数据结构 Servlet 接口:主要定义了servlet的生命周期方法 ServletConfig接口:为servlet提供了使用容器服务的若干重要对象和方法. ServletCon ...
- jquery +/-小样式
<script>部分 var num = 0; $(document).on('click','#add',function(){ _this = $(this); div = _this ...
- JVM的内存分区
JVM的内存分区 这篇文章尝试讨论清楚JVM的内存分区情况. 1. JVM的内存和系统内存的关系 下图是对系统内存及JVM内存的大致描绘 对大多数操作系统,内存可以分为物理内存RAM及Sw ...
- AngularJS执行流程详解(转)
一.启动阶段 大家应该都知道,当浏览器加载一个HTML页面时,它会将HMTL页面先解析成DOM树,然后逐个加载DOM树中的每一个元素节点.我们可以把AngularJS当做一个类似jQuery的js库, ...
- windows Apache服务器配置
Apache 64位可以而32位不可以 安装Apache服务 注意: 如果没有自己设置Apache服务名,后面都可不跟-n "服务名",即采用默认的服务名称. 必须用管理员提示符打 ...
- 小白入门Docker基础篇
docker是什么 百科上的解释是这样的: Docker 是一个开源的应用容器引擎,让开发者可以打包他们的应用以及依赖包到一个可移植的容器中,然后发布到任何流行的 Linux 机器上,也可以实现虚拟化 ...
- tensorflow.python.framework.errors_impl.OutOfRangeError: FIFOQueue
tensorflow.python.framework.errors_impl.OutOfRangeError: FIFOQueue 原创文章,请勿转载哦~!! 觉得有用的话,欢迎一起讨论相互学习~F ...
- HDU [P1704] Rank
传递闭包裸题 但是本题的Floyd一定要优化,不然会T cpp #include <iostream> #include <cstdio> #include <cstri ...
- 洛谷 [P1169] [ZJOI2007] 最大的正方形
本题是一道求最大子矩阵的题,可以使用悬线法来做,因为是相邻的01矩阵,所以需要对悬线法进行改动. #include <iostream> #include <cstdio> # ...