题目链接:hdu_5193_Go to movies Ⅱ

题意:

有n个人站成一排,每个人的身高为Hi。每次有人加入或者有人离开,就要判断有多少人站反了(i < j&&Hi>Hj) 
第一行n,m,接下来n个整数(n,m<=2e4) 
接下来m行, 
0 x y 表示有一个身高为y的人插在x后面,x=0表示插在最前面。(1≤y≤n) 
1 x 表示第x个人(从左到右)离开。

题解:

官方题解:

添加或者删除一个元素时,维护逆序对时,需要知道在它之前有多少个数比它大,在它之后有多少个数比他小。有下标和权值两个维度,可以使用两个数据结构嵌套。题目中n=20000,范围不大,外层可以使用分块维护下标,这样添加和删除元素的时候, 
也很方便,直接暴力。查找权值个数时,使用树状数组比较方便。内层通过树状数组维护权值。设每块的大小为S,那么删除或者添加元素时,维护逆序对数的复杂度是O(S+P∗logn),S是块内直接暴力更新逆序对的代价, 
​P/S∗logn在前面块找比它大和在后面块中找比它小的代价,P表示当前元素的个数。为了使这两部分复杂度尽量均摊让S=P/S∗logn,S取根号Plogn 
​。直接通过分块暴力添加和删除时,块的大小会退化,需要重构,。重构一次的复杂度是S*logn, 总的重构复杂度是,与添加和删除总的复杂度一至。因此整个问题的复杂度为O(m√nlogn).

 #include<bits/stdc++.h>
#define F(i,a,b) for(int i=a;i<=b;i++)
using namespace std; const int N=,sqr=;
int ans,tot,n,flag,op,x,y,m; inline void add(int *c,int x,int v){while(x<=n)c[x]+=v,x+=x&-x;} inline int sum(int *c,int x){int an=;while(x>)an+=c[x],x-=(x&-x);return an;} struct node
{
int dt[sqr+],sz,c[N],nxt,pre;
void init(){memset(c,,sizeof(c)),sz=,pre=,nxt=;}
}b[]; int getpos(int &x)
{
int i=;
while(b[i].sz<x&&b[i].nxt)x-=b[i].sz,i=b[i].nxt;
return i;
} int cal(int x)
{
int an=;
int id=getpos(x);
F(i,,x-)if(b[id].dt[i]>b[id].dt[x])an++;
F(i,x+,b[id].sz)if(b[id].dt[i]<b[id].dt[x])an++;
for(int i=;i!=id;i=b[i].nxt)
an+=sum(b[i].c,n)-sum(b[i].c,b[id].dt[x]);
for(int i=b[id].nxt;i!=;i=b[i].nxt)
an+=sum(b[i].c,b[id].dt[x]-);
return an;
} void del(int x){
ans-=cal(x);
int id=getpos(x);
add(b[id].c,b[id].dt[x],-);
F(i,x+,b[id].sz)b[id].dt[i-]=b[id].dt[i];
b[id].sz--;
if(b[id].sz==)//删除块
{
int pre=b[id].pre,net=b[id].nxt;
b[net].pre=pre,b[pre].nxt=net;
}
} void ins(int x,int y)
{
int xx=x;
if(flag==)
{
b[].init(),b[].dt[++b[].sz]=y;
add(b[].c,b[].dt[],),flag=;
return;
}
int id=getpos(x);//在第几个块
if(b[id].sz==sqr)//块分裂
{
b[++tot].init(),b[id].sz++;
int cnt=;
for(int i=b[id].sz;i>x;i--)
b[id].dt[i]=b[id].dt[i-];
b[id].dt[x]=y;
add(b[id].c,b[id].dt[x],);
int len=b[id].sz/;
F(i,len+,b[id].sz)
{
b[tot].dt[++cnt]=b[id].dt[i];
add(b[tot].c,b[id].dt[i],);
add(b[id].c,b[id].dt[i],-);
}
b[tot].sz=cnt,b[tot].nxt=b[id].nxt;
b[id].nxt=tot,b[id].sz=len,b[tot].pre=id;
}
else{
b[id].sz++;
for(int i=b[id].sz;i>x;i--)
b[id].dt[i]=b[id].dt[i-];
b[id].dt[x]=y;
add(b[id].c,b[id].dt[x],);
}
ans+=cal(xx);
} int main(){
while(~scanf("%d%d",&n,&m))
{
ans=,tot=,b[].sz=,flag=;
F(i,,n)scanf("%d",&y),ins(i,y);
F(i,,m)
{
scanf("%d",&op);
if(op==)scanf("%d%d",&x,&y),ins(x+,y);
else scanf("%d",&x),del(x);
printf("%d\n",ans);
}
}
return ;
}

hdu_5193_Go to movies Ⅱ(带插入删除的逆序对,块状链表)的更多相关文章

  1. [BZOJ 3295] [luogu 3157] [CQOI2011]动态逆序对(树状数组套权值线段树)

    [BZOJ 3295] [luogu 3157] [CQOI2011] 动态逆序对 (树状数组套权值线段树) 题面 给出一个长度为n的排列,每次操作删除一个数,求每次操作前排列逆序对的个数 分析 每次 ...

  2. BZOJ 3295: [Cqoi2011]动态逆序对 [CDQ分治]

    RT 传送门 首先可以看成倒着插入,求逆序对数 每个数分配时间(注意每个数都要一个时间)$t$,$x$位置,$y$数值 $CDQ(l,r)$时归并排序$x$ 然后用$[l,mid]$的加入更新$[mi ...

  3. [BZOJ3295] [Cqoi2011]动态逆序对(带修改主席树)

    题目描述 对于序列A,它的逆序对数定义为满足i<j,且Ai>Aj的数对(i,j)的个数.给1到n的一个排列,按照某种顺序依次删除m个元素,你的任务是在每次删除一个元素之前统计整个序列的逆序 ...

  4. 【BZOJ】3065: 带插入区间K小值

    http://www.lydsy.com/JudgeOnline/problem.php?id=3065 题意:带插入.修改的区间k小值在线查询.(原序列n<=35000, 询问<=175 ...

  5. 「C语言」单链表/双向链表的建立/遍历/插入/删除

    最近临近期末的C语言课程设计比平时练习作业一下难了不止一个档次,第一次接触到了C语言的框架开发,了解了View(界面层).Service(业务逻辑层).Persistence(持久化层)的分离和耦合, ...

  6. python Trie树和双数组TRIE树的实现. 拥有3个功能:插入,删除,给前缀智能找到所有能匹配的单词

    #coding=utf- #字典嵌套牛逼,别人写的,这样每一层非常多的东西,搜索就快了,树高26.所以整体搜索一个不关多大的单词表 #还是O(). ''' Python 字典 setdefault() ...

  7. 2018.07.01 BZOJ3295: [Cqoi2011]动态逆序对(带修主席树)

    3295: [Cqoi2011]动态逆序对 **Time Limit: 10 Sec Memory Limit: 128 MB Description 对于序列A,它的逆序对数定义为满足i<j& ...

  8. [BZOJ3065]带插入区间K小值 解题报告 替罪羊树+值域线段树

    刚了一天的题终于切掉了,数据结构题的代码真**难调,这是我做过的第一道树套树题,做完后感觉对树套树都有阴影了......下面写一下做题记录. Portal Gun:[BZOJ3065]带插入区间k小值 ...

  9. [LeetCode] Insert Delete GetRandom O(1) - Duplicates allowed 常数时间内插入删除和获得随机数 - 允许重复

    Design a data structure that supports all following operations in average O(1) time. Note: Duplicate ...

随机推荐

  1. html5之datalist标签

    当我看到这个标签的时候,其实我是很愤怒的.因为我以前实现过这个标签的功能,当时是无比的费劲.什么js库呀,function呀.我靠,统统去屎吧,哥有datalist了.那种感觉就好像自己千辛万苦去追去 ...

  2. 转:Monoids and Finger Trees

    转自:http://apfelmus.nfshost.com/articles/monoid-fingertree.html This post grew out of the big monoid ...

  3. leetcode 024

    Given a linked list, swap every two adjacent nodes and return its head. For example,Given 1->2-&g ...

  4. [转]学好Mac常用命令,助力iOS开发

    转自:http://www.jianshu.com/p/d9ec00d28237   序言 在iOS开发的过程中,更多地注重iOS开发的效率,熟练使用Mac终端操作的常用命令,可以让你更好的游刃于iO ...

  5. 记JavaScript的入门学习(一)

    2016年11月20日算是每天都会利用点时间去思考,去学习,接下来便是我学习JavaScript的前前后后,希望早日学完JavaScript和jQuery 之前学校毕业设计选择了一个如下图的题目 希望 ...

  6. json字符串和对象的相互转化

    json在代码中是经常用到的,在此总结一下json字符串和对象及数组之间的相互转化: 1.javascript函数方式: <1> JSON.stringify :把一个对象转换成json字 ...

  7. 设计模式 -- 组合模式 (Composite Pattern)

    定义: 对象组合成部分整体结构,单个对象和组合对象具有一致性. 看了下大概结构就是集团总公司和子公司那种层级结构. 角色介绍: Component :抽象根节点:其实相当去总公司,抽象子类共有的方法: ...

  8. 分布式存储 CentOS6.5虚拟机环境搭建FastDFS-5.0.5集群(转载-2)

    原文:http://www.cnblogs.com/PurpleDream/p/4510279.html 分布式存储 CentOS6.5虚拟机环境搭建FastDFS-5.0.5集群 前言:       ...

  9. 【IIS】windows2008 ii7 设置访问网站提示帐号密码登录

    3个步骤: 1.添加windows身份验证: windows2008默认是不启用的,需要我们自己去启动,在管理工具 - 服务器管理- 角色 ,拉下去,下面有个[添加角色服务],安全性- Windows ...

  10. android弹出dialog后,activity得到焦点

    1.getWindow().setFlags(WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE,WindowManager.LayoutParams.FLAG ...