noip2017 D2T3 列队 某zz选手当时直接放弃了写了50还写错了

题目大意:

有一个n行m列的方阵,第i行j列的点编号为(i-1)m+j

每次把第x行y列的点拿出来,然后把这一行它之后的点都向左推,把最后一列x行之后的点都向上推

然后把拿出来的点放到最后一个位置,询问这个点的编号

思路:

因为n+q并不大 可以动态开点每行维护一个权值线段树 并对最后一行单独维护

对于新进来的点使用vector维护

 #include<iostream>
#include<cstdio>
#include<cmath>
#include<cstdlib>
#include<cstring>
#include<algorithm>
#include<vector>
#include<queue>
#define inf 2139062143
#define ll long long
#define MAXN 300100
using namespace std;
inline int read()
{
int x=,f=;char ch=getchar();
while(!isdigit(ch)) {if(ch=='-') f=-;ch=getchar();}
while(isdigit(ch)) {x=x*+ch-'';ch=getchar();}
return x*f;
}
int n,m,q,rt[MAXN],ls[MAXN*],rs[MAXN*],sum[MAXN*],sz;
ll ans[MAXN<<];
vector <ll> vec[MAXN];
int query(int &k,int l,int r,int x)
{
if(!k) k=++sz,sum[k]=r-l+;
sum[k]--;
if(l==r) return l;
int mid=l+r>>,sl=ls[k]?sum[ls[k]]:mid-l+;
if(x<=sl) query(ls[k],l,mid,x);
else query(rs[k],mid+,r,x-sl);
}
int main()
{
n=read(),m=read(),q=read();int a,b,res;
for(int i=;i<=n;i++) ans[i]=1ll*i*m;
for(int i=n+;i<=n+q;i++)
{
a=read(),b=read();
if(b==m) ans[i]=ans[query(rt[],,n+q,a)];
else
{
res=query(rt[a],,m+q,b);
if(res<m) ans[i]=(a-1ll)*m*1ll+res;
else ans[i]=vec[a][res-m];
vec[a].push_back(ans[query(rt[],,n+q,a)]);
}
printf("%lld\n",ans[i]);
}
}
05-11 17:23