树套树:

约等于是个暴力了。以区间线段树的方式开一棵权值线段树,在权值线段树的每一个点上以动态开点的方式开一棵区间线段树。

结果非常惨烈(时限20s)

#include<iostream>
#include<cstdio>
#include<algorithm>
using namespace std;
const int N=23000005,M=500005;
int n,m,tot,rt[M],sz,lz[N];
long long sum[N],mp[M];
struct data
{
int l,r;
}tr[N];
struct node
{
int a,b,op;
long long c;
}ans[M];
int find(long long x)
{
return lower_bound(mp+1,mp+tot+1,x)-mp-1;
}
void pushdown(int x,int l,int r)
{
int mid=(l+r)/2;
if(!tr[x].l)
tr[x].l=++sz;
if(!tr[x].r)
tr[x].r=++sz;
int a=tr[x].l,b=tr[x].r;
lz[a]+=lz[x];
lz[b]+=lz[x];
sum[a]+=(long long)(mid-l+1)*(long long)lz[x];
sum[b]+=(long long)(r-mid)*(long long)lz[x];
lz[x]=0;
}
void change(int &ro,int l,int r,int ll,int rr)
{
if(!ro)
ro=++sz;
if(lz[ro])
pushdown(ro,l,r);
if(l>=ll&&r<=rr)
{
sum[ro]+=(long long)(r-l+1);
lz[ro]++;
return;
}
int mid=(l+r)/2;
if(ll<=mid)
change(tr[ro].l,l,mid,ll,rr);
if(rr>mid)
change(tr[ro].r,mid+1,r,ll,rr);
sum[ro]=(long long)sum[tr[ro].l]+sum[tr[ro].r];
}
void update(int a,int b,int c,int now,int l,int r)
{
change(rt[now],1,n,a,b);
if(l==r)
return;
int mid=(l+r)/2;
if(c<=mid)
update(a,b,c,now<<1,l,mid);
else
update(a,b,c,now<<1|1,mid+1,r);
}
long long query(int k,int l,int r,int ll,int rr)
{
if(lz[k])
pushdown(k,l,r);
if (l>=ll&&r<=rr)
return sum[k];
int mid=(l+r)/2;
long long ans=0;
if(ll<=mid)
ans+=query(tr[k].l,l,mid,ll,rr);
if(rr>mid)
ans+=query(tr[k].r,mid+1,r,ll,rr);
return ans;
}
int ques(int a,int b,long long c)
{
int l=1,r=tot,k=1;
while(l!=r)
{
int mid=(l+r)>>1;
long long t=query(rt[k<<1],1,n,a,b);
if(t>=c)
r=mid,k<<=1;
else
l=mid+1,k=k<<1|1,c-=(long long)t;
}
return l;
}
int main()
{
scanf("%d%d",&n,&m);
for (int i=1;i<=m;i++)
{
scanf("%d%d%d%lld",&ans[i].op,&ans[i].a,&ans[i].b,&ans[i].c);
if(ans[i].op==1)
mp[++tot]=ans[i].c;
}
sort(mp+1,mp+tot+1);
tot=unique(mp+1,mp+tot+1)-mp-1;
for(int i=1;i<=m;i++)
if(ans[i].op==1)
{
int k=find(ans[i].c)+1;
update(ans[i].a,ans[i].b,k,1,1,tot);
}
else
{
long long cnt=query(rt[1],1,n,ans[i].a,ans[i].b);
printf("%lld\n",mp[ques(ans[i].a,ans[i].b,cnt-ans[i].c+1)]);
}
return 0;
}

整体二分

首先我一开始没有注意到的:

1操作中abs(c)<=N

2操作中c<=Maxlongint

也就是说答案在0到n之间,那么久可以整体二分来做了

维护一棵区间线段树,维护区间加标记和区间清零标记。

每次操作前清零

设当前答案区间为[l,r],询问区间为[x,y],mid=(l+r)>>1;

对于插入操作:如果插入的数>mid,就在其区间上+1,放到前一堆(我不知道怎么形容了,否则不操作,放倒后一堆;

对于查询操作:如果查询区间内的数符合k个要求,放到前一堆,否则放到后一堆。

直到l==r,更新[x,y]内的答案

写丑了

#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<cmath>
using namespace std;
const int N=500005;
int n,m,ans[N];
long long sum[N*4],lz[N*4],tg[N*4];
struct qwe
{
int l,r,id,n,p,x;
}a[N];
int read()
{
int r=0,f=1;
char p=getchar();
while(p>'9'||p<'0')
{
if(p=='-')
f=-1;
p=getchar();
}
while(p>='0'&&p<='9')
{
r=r*10+p-48;
p=getchar();
}
return r*f;
}
void clear(int now)
{
sum[now]=lz[now]=0;
tg[now]=1;
}
bool cmp(const qwe &a,const qwe &b)
{
return a.n<b.n;
}
void pd(int now,int l,int r)
{
if(tg[now])
{
clear(now<<1);
clear(now<<1|1);
tg[now]=0;
}
int mid=(l+r)/2;
if(lz[now])
{
lz[now<<1]+=lz[now];
lz[now<<1|1]+=lz[now];
sum[now<<1]+=lz[now]*(mid-l+1);
sum[now<<1|1]+=lz[now]*(r-mid);
lz[now]=0;
}
}
void ud(int now)
{
sum[now]=sum[now<<1]+sum[now<<1|1];
}
void update(int now,int l,int r,int ll,int rr)
{
if(ll<=l&&r<=rr)
{
sum[now]+=(r-l+1);
lz[now]++;
return;
}
int mid=(l+r)/2;
pd(now,l,r);
if(ll<=mid)
update(now<<1,l,mid,ll,rr);
if(rr>mid)
update(now<<1|1,mid+1,r,ll,rr);
ud(now);
}
long long ques(int now,int l,int r,int ll,int rr)
{
if(ll<=l&&r<=rr)
return sum[now];
int mid=(l+r)/2;
pd(now,l,r);
long long ans=0;
if(ll<=mid)
ans+=ques(now<<1,l,mid,ll,rr);
if(rr>mid)
ans+=ques(now<<1|1,mid+1,r,ll,rr);
return ans;
}
void erfen(int l,int r,int x,int y)
{
if(l==r)
{
for(int i=x;i<=y;i++)
if(a[i].p==2)
ans[a[i].id]=l;
return;
}
int mid=(l+r)/2,pl=0,pr=y-x+1;
clear(1);
for(int i=x;i<=y;i++)
if(a[i].p==1)
{
if(a[i].x<=mid)
a[i].n=++pl;
else
{
update(1,1,n,a[i].l,a[i].r);
a[i].n=++pr;
}
}
else
{
long long t=ques(1,1,n,a[i].l,a[i].r);
if(t>=a[i].x)
a[i].n=++pr;
else
{
a[i].x=a[i].x-t;
a[i].n=++pl;
}
}
sort(a+x,a+y+1,cmp);
erfen(l,mid,x,x+pl-1);
erfen(mid+1,r,x+pl,y);
}
int main()
{
n=read(),m=read();
for(int i=1;i<=m;i++)
a[i].p=read(),a[i].l=read(),a[i].r=read(),a[i].x=read(),a[i].id=i;
erfen(0,n,1,m);
for(int i=1;i<=m;i++)
if(ans[i])
printf("%d\n",ans[i]);
return 0;
}

bzoj 3110 [Zjoi2013]K大数查询【树套树||整体二分】的更多相关文章

  1. BZOJ 3110: [Zjoi2013]K大数查询 [树套树]

    3110: [Zjoi2013]K大数查询 Time Limit: 20 Sec  Memory Limit: 512 MBSubmit: 6050  Solved: 2007[Submit][Sta ...

  2. 树套树专题——bzoj 3110: [Zjoi2013] K大数查询 &amp; 3236 [Ahoi2013] 作业 题解

    [原题1] 3110: [Zjoi2013]K大数查询 Time Limit: 20 Sec  Memory Limit: 512 MB Submit: 978  Solved: 476 Descri ...

  3. bzoj 3110: [Zjoi2013]K大数查询 树状数组套线段树

    3110: [Zjoi2013]K大数查询 Time Limit: 20 Sec  Memory Limit: 512 MBSubmit: 1384  Solved: 629[Submit][Stat ...

  4. BZOJ 3110: [Zjoi2013]K大数查询( 树状数组套主席树 )

    BIT+(可持久化)权值线段树, 用到了BIT的差分技巧. 时间复杂度O(Nlog^2(N)) ---------------------------------------------------- ...

  5. BZOJ 3110([Zjoi2013]K大数查询-区间第k大[段修改,在线]-树状数组套函数式线段树)

    3110: [Zjoi2013]K大数查询 Time Limit: 20 Sec   Memory Limit: 512 MB Submit: 418   Solved: 235 [ Submit][ ...

  6. BZOJ 3110 [Zjoi2013]K大数查询(整体二分)

    3110: [Zjoi2013]K大数查询 Time Limit: 20 Sec  Memory Limit: 512 MBSubmit: 11654  Solved: 3505[Submit][St ...

  7. BZOJ 3110 [Zjoi2013]K大数查询 ——树套树

    [题目分析] 外层区间线段树,内层是动态开点的权值线段树. SY神犇说树套树注重的是内外层的数据结构的选择问题,果然很重要啊. 动态开点的实现方法很好. [代码] #include <cstdi ...

  8. BZOJ 3110: [Zjoi2013]K大数查询 [整体二分]

    有N个位置,M个操作.操作有两种,每次操作如果是1 a b c的形式表示在第a个位置到第b个位置,每个位置加入一个数c如果是2 a b c形式,表示询问从第a个位置到第b个位置,第C大的数是多少. N ...

  9. [BZOJ 3110] [Zjoi2013] K大数查询 【树套树】

    题目链接: BZOJ - 3110 题目分析 这道题是一道树套树的典型题目,我们使用线段树套线段树,一层是区间线段树,一层是权值线段树.一般的思路是外层用区间线段树,内层用权值线段树,但是这样貌似会很 ...

随机推荐

  1. linux 报错:E: Package 'libmemcached' has no installation candidate

    linux 报错:E: Package 'libmemcached' has no installation candidate 网上查资料说是软件安装源没有这个软件,需要添加软件源. 1.备份源列表 ...

  2. POJ 1094 Sorting It All Out【拓扑排序】

    题目链接: http://poj.org/problem?id=1094 题意: 给定前n个字母的大小关系,问你是否 根据前xxx个关系得到上升序列 所有关系都无法确定唯一的一个序列 第xxx个关系导 ...

  3. 洛谷——P1265 公路修建

    P1265 公路修建 题目描述 某国有n个城市,它们互相之间没有公路相通,因此交通十分不便.为解决这一“行路难”的问题,政府决定修建公路.修建公路的任务由各城市共同完成. 修建工程分若干轮完成.在每一 ...

  4. 2017多校Round2(hdu6045~hdu6055)

    补题进度:10/11 1001(不等式) 根据题意列不等式,解一解就行了 1002(套路) 题意: 给定一个随机产生的1e6*1e6的矩阵和一个1e3*1e3的矩阵,你要回答这个1e3*1e3的小矩阵 ...

  5. mysql 统计数据,按照日期分组,把没有数据的日期也展示出来

    因为业务需求,要统计每天的新增用户并且要用折线图的方式展示. 如果其中有一天没有新增用户的话,这一天就是空缺的,在绘制折线图的时候是不允许的,所有要求把没有数据的日期也要在图表显示. 查询2019-0 ...

  6. 分享一下然让显卡满血复活的小技巧(GTX)

    分享一下然让显卡满血复活的小技巧 笔者在玩大型游戏卡顿15fps下载如下操作 GTX950玩大型游戏都不会卡帧率稳定在30fps 下载GeForce Experience下载更新最新驱动 下载如下程序 ...

  7. Windows下SVN服务器及客户端的使用

    原文地址:windows下配置VisualSVN Server服务器 作者:Deem_passion 下载安装文件: 服务端安装文件:VisualSVN-Server-1.6.2 客户端安装文件:To ...

  8. Fragment实践之聊天窗体

    前几天刚学了android的fragment,总是停留在简单的demo,也许永远都学不会. 今天,我要动手向我的聊天软件开刀.今天.用Fragment来实现一个例如以下图效果的聊天界面. waterm ...

  9. 如何在 Linux 环境下配置 Nagios Remote Plugin Executor (NRPE)

    为 NRPE 配置自定义命令 远程服务器上安装 下面列出了一些可以用于 NRPE 的自定义命令.这些命令在远程服务器的 /etc/nagios/nrpe.cfg 文件中定义. ## 当 1.5.15 ...

  10. 【原创】PHP扩展开发入门

    PHP扩展开发入门 作者:wf (360电商技术组) 在我们编写自己的第一个php扩展之前,先了解一下php的总体架构和执行机制. php的架构如图1所看到的. 当中一个重要的就是SAPI(serve ...