ZOJ 2112 Dynamic Rankings(树状数组+主席树)
题意
\(n\) 个数,\(m\) 个操作,每次操作修改某个数,或者询问某个区间的第 \(K\) 小值。
\(1 \leq n \leq 50000\)
\(1 \leq m \leq 10000\)
思路
动态区间第 \(K\) 小的模板题。回到主席树的职能,它维护了一排前缀线段树,假如我们要快速求一段区间的总和,显然优先考虑前缀和;然而如果带上修改,就要考虑树状数组了。这里也相同,我们只需用树状数组套主席树即可,\(x\) 号主席树维护的不再是 \([1,x]\) 的信息,而是 \([x-\text{lowbit(x)}+1,x]\) 的信息。
修改时,像树状数组的修改一样不断加 \(\text{lowbit}\) ,修改 \(\log\) 棵主席树。
查询时,像树状数组一样不断减 \(\text{lowbit}\) ,查询 \(2\log\) 棵主席树。这里可以传入两个 \(\text{vector}\) ,一个加、一个减,然后就可以轻松求出区间第 \(K\) 值了。
总复杂度 \((n+m)\log^2n\) ,然而过不了。
这道题比较卡常,初始的序列应该再用一棵静态主席树解决,动态主席树只存修改的信息,查询时额外加入静态主席树上要加的位置和要减的位置即可。
总复杂度 \(m\log^2n\) ,现在可以过了。
代码
#include<bits/stdc++.h>
#define FOR(i,x,y) for(int i=(x),i##END=(y);i<=i##END;++i)
#define DOR(i,x,y) for(int i=(x),i##END=(y);i>=i##END;--i)
typedef long long LL;
using namespace std;
const int N=5e4+5;
const int M=1e4+5;
const int NN=2e6+5;
struct ChairmanTree
{
int lson[NN],rson[NN],cnt[NN];
int rt[N+N],tot;
int &operator [](const int x){return rt[x];}
void build()
{
memset(rt,0,sizeof(rt));
cnt[tot=0]=0,lson[0]=rson[0]=0;
}
void create(int &k){cnt[++tot]=cnt[k],lson[tot]=lson[k],rson[tot]=rson[k],k=tot;}
void update(int &k,int x,int val,int l,int r)
{
create(k);
if(l==r)
{
cnt[k]+=val;
return;
}
int mid=(l+r)>>1;
if(x<=mid)update(lson[k],x,val,l,mid);
else update(rson[k],x,val,mid+1,r);
cnt[k]=cnt[lson[k]]+cnt[rson[k]];
}
int query(vector<int>&A,vector<int>&B,int K,int l,int r)
{
if(l==r)return l;
int mid=(l+r)>>1,Cnt=0;
FOR(i,0,(int)A.size()-1)Cnt+=cnt[lson[A[i]]];
FOR(i,0,(int)B.size()-1)Cnt-=cnt[lson[B[i]]];
if(Cnt>=K)
{
FOR(i,0,(int)A.size()-1)A[i]=lson[A[i]];
FOR(i,0,(int)B.size()-1)B[i]=lson[B[i]];
return query(A,B,K,l,mid);
}
else
{
FOR(i,0,(int)A.size()-1)A[i]=rson[A[i]];
FOR(i,0,(int)B.size()-1)B[i]=rson[B[i]];
return query(A,B,K-Cnt,mid+1,r);
}
}
}CT;
int a[N],L[M],R[M],K[M],disc[N+M];
int n,m,tot;
#define lowbit(x) ((x)&-(x))
void update(int k,int val)
{
int _val=a[k];
a[k]=val;
for(;k<=n;k+=lowbit(k))
{
CT.update(CT[k],_val,-1,1,tot);
CT.update(CT[k],val,1,1,tot);
}
}
int query(int l,int r,int k)
{
vector<int>A,B;
A.clear(),B.clear();
A.push_back(CT[N+r]),B.push_back(CT[N+l-1]);
for(;r>0;r^=lowbit(r))A.push_back(CT[r]);
for(l--;l>0;l^=lowbit(l))B.push_back(CT[l]);
return CT.query(A,B,k,1,tot);
}
#undef lowbit
int main()
{
int T;
scanf("%d",&T);
while(T--)
{
CT.build();
tot=0;
scanf("%d%d",&n,&m);
FOR(i,1,n)scanf("%d",&a[i]),disc[++tot]=a[i];
FOR(i,1,m)
{
char str[5];
scanf("%s",str);
if(str[0]=='Q')scanf("%d%d%d",&L[i],&R[i],&K[i]);
else scanf("%d%d",&L[i],&R[i]),K[i]=-1,disc[++tot]=R[i];
}
sort(disc+1,disc+1+tot);
tot=unique(disc+1,disc+1+tot)-disc-1;
FOR(i,1,n)a[i]=lower_bound(disc+1,disc+1+tot,a[i])-disc;
FOR(i,1,m)if(K[i]==-1)R[i]=lower_bound(disc+1,disc+1+tot,R[i])-disc;
FOR(i,1,n)CT.update(CT[N+i]=CT[N+i-1],a[i],1,1,tot);
FOR(i,1,m)
{
if(K[i]!=-1)printf("%d\n",disc[query(L[i],R[i],K[i])]);
else update(L[i],R[i]);
}
}
return 0;
}
ZOJ 2112 Dynamic Rankings(树状数组+主席树)的更多相关文章
- ZOJ - 2112 Dynamic Rankings(BIT套主席树)
纠结了好久的一道题,以前是用线段树套平衡树二分做的,感觉时间复杂度和分块差不多了... 终于用BIT套函数式线段树了过了,120ms就是快,此题主要是卡内存. 假设离散后有ns个不同的值,递归层数是l ...
- BZOJ_1901_Zju2112 Dynamic Rankings_树状数组+主席树
BZOJ_1901_Zju2112 Dynamic Rankings_树状数组+主席树 题意: 给定一个含有n个数的序列a[1],a[2],a[3]……a[n],程序必须回答这样的询问:对于给定的i, ...
- zoj2112 树状数组+主席树 区间动第k大
Dynamic Rankings Time Limit: 10000MS Memory Limit: 32768KB 64bit IO Format: %lld & %llu Subm ...
- 【bzoj1146】[CTSC2008]网络管理Network 倍增LCA+dfs序+树状数组+主席树
题目描述 M公司是一个非常庞大的跨国公司,在许多国家都设有它的下属分支机构或部门.为了让分布在世界各地的N个部门之间协同工作,公司搭建了一个连接整个公司的通信网络.该网络的结构由N个路由器和N-1条高 ...
- 【bzoj3744】Gty的妹子序列 分块+树状数组+主席树
题目描述 我早已习惯你不在身边, 人间四月天 寂寞断了弦. 回望身后蓝天, 跟再见说再见…… 某天,蒟蒻Autumn发现了从 Gty的妹子树(bzoj3720) 上掉落下来了许多妹子,他发现 她们排成 ...
- BZOJ_2120_数颜色_Set+树状数组+主席树
BZOJ_2120_数颜色_Set+树状数组+主席树 Description 墨墨购买了一套N支彩色画笔(其中有些颜色可能相同),摆成一排,你需要回答墨墨的提问.墨墨会像你发布如下指令: 1. Q L ...
- P1972 [SDOI2009]HH的项链[离线+树状数组/主席树/分块/模拟]
题目背景 无 题目描述 HH 有一串由各种漂亮的贝壳组成的项链.HH 相信不同的贝壳会带来好运,所以每次散步完后,他都会随意取出一段贝壳,思考它们所表达的含义.HH 不断地收集新的贝壳,因此,他的项链 ...
- ZOJ 2112 Dynamic Rankings(树状数组+主席树)
The Company Dynamic Rankings has developed a new kind of computer that is no longer satisfied with t ...
- [luogu2617][bzoj1901][Zju2112]Dynamic Rankings【树套树+树状数组+主席树】
题目网址 [传送门] 题目大意 请你设计一个数据结构,支持单点修改,区间查询排名k. 感想(以下省略脏话inf个字) 真的强力吹爆洛谷数据,一般的树套树还给我T了一般的点,加强的待修主席树还给我卡了几 ...
随机推荐
- 压缩和解压缩(I)
ZipArchive 压缩方法 -(void)zipArchiveWithFiles { //创建解压缩对象 ZipArchive *zip = [[ZipArchive alloc]init]; / ...
- 文件格式(图像 IO 14.3)
文件格式 图片加载性能取决于加载大图的时间和解压小图时间的权衡.很多苹果的文档都说PNG是iOS所有图片加载的最好格式.但这是极度误导的过时信息了. PNG图片使用的无损压缩算法可以比使用JPEG的图 ...
- 【Hive学习之八】Hive 调优【重要】
环境 虚拟机:VMware 10 Linux版本:CentOS-6.5-x86_64 客户端:Xshell4 FTP:Xftp4 jdk8 hadoop-3.1.1 apache-hive-3.1.1 ...
- 财务自由VS精神自由
财务自由 财务自由,在物质层面改善人的生活.它使人不愁生计.住更宽敞明亮的房间,穿锦衣绸缎,自由自在地游玩,做自己想做的事儿.可是,这就是它的能力所及了.钱无法改变人的品味.审美和人格.它也无法告诉人 ...
- winscp 怎么用私钥文件登录的,以.ppk结尾的密钥文件
Winscp默认用帐号密码登录,用私钥文件登录需要在高级选项里面的SSH--验证里面选择文件. Winscp使用的是putty作为SSH登录工具,而puttygen所生成的是以.ppk结尾的密钥文件. ...
- Codeforce 513A - Game
Two players play a simple game. Each player is provided with a box with balls. First player's box co ...
- Step1:SQL Server 复制介绍
一.本文所涉及的内容(Contents) 本文所涉及的内容(Contents) 前言(Introduction) 复制逻辑结构图(Construction) 系列文章索引(Catalog) 总结&am ...
- python requests接口测试
Python 标准库中的 urllib2 模块提供了你所需要的大多数 HTTP 功能,但是它的 API 太渣了.它是为另一个时代.另一个互联网所创建的.它需要巨量的工作,甚至包括各种方法覆盖,来完成最 ...
- 一个讲课截屏 清明DAY2
灰常混乱 放弃吧........ 不断做平方差公式 到i时,前面已经求出之前数字的逆元了 r是一个比i小的数 第四行×i,r 的逆元 BSGS 暴力枚举枚举到Φ(m)个
- centos系统swap设置 查看swap分区的方法
交换分区swap,意思是“交换”.“实物交易”,它的功能就是在内存不够的情况下,操作系统先把内存中暂时不用的数据,存到硬盘的交换空间,腾出内存来让别的程序运行,和Windows的虚拟内存(pagefi ...