zoukankan      html  css  js  c++  java
  • [SHOI2013]发牌 解题报告

    [SHOI2013]发牌

    题意

    对一个(1sim n(nle 7 imes 10^5))的环,指标最开始在(1),每次删去顺时针往后第(d_i)个元素,指标移到下一个位置。要求输出每次删去元素的标号。


    看了沙茶数据范围,我就放弃了平衡树,毕竟我这种人丑常数大的选手?

    然后开始思考线段树或者树状数组,yy了好一会儿才想出一个沙茶的做法,写了以后发现题解好像没有我这么写的,就来口胡一下。

    在线段树上维护每个点左边还有多少个元素存在。

    每次询问的时候,有一个上次开始的删去的位置(s)(初始为(0)),然后统计一下(s)前面有多少个元素,判断这次删掉的是在(s)左边还是右边。

    假设每次输入的是(d),如果在(s)左边,就(d-)右边元素个数,在右边就(d+)左边元素个数,然后直接在线段树上二分找就可以了。


    Code:

    #include <cstdio>
    #include <cctype>
    #include <algorithm>
    using std::max;
    const int N=7e5+10;
    int read()
    {
    	int x=0;char c=getchar();
    	while(!isdigit(c)) c=getchar();
    	while(isdigit(c)) x=x*10+c-'0',c=getchar();
    	return x;
    }
    int sum[N],n,s;
    void add(int x){while(x<=n)++sum[x],x+=x&-x;}
    int ask(int x){int ret=0;while(x)ret+=sum[x],x-=x&-x;return ret;}
    int mx[N<<2],tag[N<<2];
    #define ls id<<1
    #define rs id<<1|1
    void build(int id,int l,int r)
    {
    	mx[id]=r;
    	if(l==r)return;
    	int mid=l+r>>1;
    	build(ls,l,mid),build(rs,mid+1,r);
    }
    void pushdown(int id)
    {
    	if(tag[id])
    	{
    		mx[ls]+=tag[id],mx[rs]+=tag[id];
    		tag[ls]+=tag[id],tag[rs]+=tag[id];
    		tag[id]=0;
    	}
    }
    void query(int id,int l,int r,int k)
    {
    	if(l==r) {printf("%d
    ",s=l),mx[id]=0,add(l);return;}
    	pushdown(id);
    	int mid=l+r>>1;
    	if(mx[ls]>=k) query(ls,l,mid,k);
    	else query(rs,mid+1,r,k);
    }
    void change(int id,int l,int r,int p)
    {
    	if(l>=p){--mx[id],--tag[id];return;}
    	pushdown(id);
    	int mid=l+r>>1;
    	if(p<=mid) change(ls,l,mid,p);
    	change(rs,mid+1,r,p);
    	mx[id]=max(mx[ls],mx[rs]);
    }
    int main()
    {
    	n=read();
    	build(1,1,n);
    	for(int rig,lef,r,i=n;i;i--)
    	{
    		r=read()%i;
    		lef=s-ask(s);//左边个数
    		rig=i-lef;//右边个数
    		if(rig<=r) r-=rig;//左边
    		else r+=lef;
    		query(1,1,n,r+1);
    		change(1,1,n,s);
    	}
    	return 0;
    }
    

    2019.2.12

  • 相关阅读:
    (转)mtr命令详解诊断网络路由
    WinDbg使用介绍
    windbg-bp、 bm、 bu、 bl、 bc、 ba(断点、硬件断点)
    【转】25.windbg-!gle、g(错误码、g系列)
    umdh windbg分析内存泄露
    windbg !logexts(自带的监控API)
    windbg cs
    windbg dds、dps、dqs
    Windbg找出memory leak的一种笨办法
    【转】windows平台多线程同步之Mutex的应用
  • 原文地址:https://www.cnblogs.com/butterflydew/p/10366657.html
Copyright © 2011-2022 走看看