数据结构题还是挺好玩的

注意到每次只变动三个点:(x,y),(x,m),(n,m),其他地方都是整块移动。

可以开n+1个线段树,前n个存每行前m-1个人,最后一个存第m列的人。

(x,y)位置的人出列时,抽出该位置的标号,加到第n+1个线段树的最后面去,抽出第n+1个线段树的第x个元素(即(x,m),加到第x个线段树的最后面去。

↑实现这一操作,可以记线段树的size,每次二分查询树上第k个元素即可确定位置。

↑当然树不可能全建出来,需要动态开点。尚未申请的结点,size直接用$ r-l+1 $计算。

写到一半想到不需要专门抽出来,只需要记录该行到该位置为止抽出过a个数,找(x,y)时实际找(x,y+a)即可。可以开个vector什么的存新加到队伍最右边的人。

这样的话更加方便&优美,而且可以用树状数组写,代码量也减小了。

但是看着自己写了一大半的代码不舍得丢,还是强行按原计划敲完了。

————

老年退役选手连NOIP题都做不来了。做这题成功遇到了数组开小,变量写混,初始化错误,输出错误等等问题,调了好久。

不开心QAQ

————

 #include<iostream>
#include<cstdio>
#define LL long long
using namespace std;
const int mxn=;
struct node{
int sz,lc,rc;
LL mk;
}t[mxn<<];
int cnt=,mod=;
int root[mxn];
int n,N,m,Q;
inline int getsz(int l,int r,int rt){
if(rt)return t[rt].sz;
return r-l+;
}
void pushup(int l,int r,int rt){
int tmp=;
int mid=(l+r)>>;
tmp+=getsz(l,mid,t[rt].lc);
tmp+=getsz(mid+,r,t[rt].rc);
t[rt].sz=tmp;
return;
}
int anspos;
int query(int k,int l,int r,int &rt){
if(!rt){
rt=++cnt;
t[rt].sz=r-l+;
}
if(l==r){
if(!t[rt].mk){//计算标号
if(mod==n+)t[rt].mk=(LL)l*m;
else t[rt].mk=(LL)(mod-)*m+l;
}
anspos=l;
return rt;
}
int mid=(l+r)>>,tmp=getsz(l,mid,t[rt].lc);
if(k<=tmp)
return query(k,l,mid,t[rt].lc);
else return query(k-tmp,mid+,r,t[rt].rc);
}
void update(LL v,int p,int l,int r,int &rt){
if(!rt){
rt=++cnt;
t[rt].sz=r-l+;
}
if(l==r){
t[rt].mk=v;
if(!v){//del
t[rt].sz=;
}
else t[rt].sz=;
return;
}
int mid=(l+r)>>;
if(p<=mid)update(v,p,l,mid,t[rt].lc);
else update(v,p,mid+,r,t[rt].rc);
pushup(l,r,rt);
return;
}
void info(int l,int r,int rt){
printf(" %d %d rt:%d sz:%d\n",l,r,rt,getsz(l,r,rt));
if(!rt)return;
if(l==r)return;
int mid=(l+r)>>;
info(l,mid,t[rt].lc);
info(mid+,r,t[rt].rc);
return;
}
int main(){
// freopen("2017phalanx.in","r",stdin);
// freopen("2017phalanx.out","w",stdout);
int i,j,x,y,tmp;
LL tmp2;
scanf("%d%d%d",&n,&m,&Q);
N=max(max(n,m),Q)<<;
while(Q--){
scanf("%d%d",&x,&y);
if(y==m){
mod=n+;
tmp=query(x,,N,root[n+]);
tmp2=t[tmp].mk;
printf("%lld\n",t[tmp].mk);
update(,anspos,,N,root[n+]);
tmp=query(n,,N,root[n+]);
update(tmp2,anspos,,N,root[n+]);
}
else{
// printf(" in\n");
mod=x;
tmp=query(y,,N,root[x]);
tmp2=t[tmp].mk;
printf("%lld\n",t[tmp].mk);
update(,anspos,,N,root[x]);//出队
// printf("anspos:%d\n",anspos);
//
mod=n+;
tmp=query(n+,,N,root[n+]);
update(tmp2,anspos,,N,root[n+]);
// printf("UPD: mk:%d pos:%d\n",tmp2,anspos);
//
tmp=query(x,,N,root[n+]);
tmp2=t[tmp].mk;
// printf("bu:%d anspos:%d\n",tmp2,anspos);
update(,anspos,,N,root[n+]);
mod=x;
tmp=query(m-,,N,root[x]);
update(tmp2,anspos,,N,root[x]);
//
}
// info(1,N,root[n+1]);
}
return ;
}

最新文章

  1. Unity3D 预设打包的注意事项
  2. SetEnvlfNoCase 记录从自己网站之外传来的请求
  3. ubuntu ulimit 设置
  4. 【css】ie6 和 ie7 下 position 与 overflow 的问题
  5. obj-m
  6. Ubuntu安装和设置SSH服务
  7. 教你21天学会C++ (有图有真相)
  8. Masonry+Infinite-Scroll实现无刷新无分页完美瀑布流(转)
  9. [tensorflow in a nutshell] tensorflow简明教程 (第一部分)
  10. 小明滚出---响应对象HttpServletResponse和请求对象HttpServletRequest实例
  11. UOJ#346. 【清华集训2017】某位歌姬的故事 动态规划
  12. html/css/js----js中遇到的一些问题
  13. [二分答案][NOIP2015]跳石头
  14. Python 文件编译为字节码的方法
  15. Python3之弹性力学——应力张量1
  16. ArcPy批量计算Mean Center的两个实例
  17. 内网访问已经启动的vue项目
  18. Mysql中use一个表出现警告:Reading table information for completion of table and column names You can turn off this feature to get a quicker startup with -A
  19. 【iCore4 双核心板_ARM】例程四:USART实验——通过命令控制LED
  20. Win7/Win10多用户同时使用远程桌面

热门文章

  1. [转帖]Intel为何吊打AMD,先进半导体工艺带来什么?
  2. .Net iTextSharp 生成pdf
  3. ZooKeeper-基础介绍
  4. git 常用命令(含删除文件)
  5. BZOJ 2742: [HEOI2012]Akai的数学作业
  6. Linux进程间通信(消息队列/信号量+共享内存)
  7. Android自定义 Dialog 对话框
  8. 解题:POI 2008 Subdivision of Kingdom
  9. bzoj 4451 : [Cerc2015]Frightful Formula FFT
  10. Linux发不出分片包的问题分析