标签:getch || nod 代码 main i++ return etc stream
题目大意:给一段空序列,每次向序列中某一个位置插入一个数,插入的位置后面所有数相应后移。
这个题比较令人头疼的是后移操作,我们不可能大面积后移。那怎么办呢?后面的人对前面有影响,那我们能不能通过离线方法,使得它变成没有影响的状态?
可以的。我们可以把输入离线,然后倒着插入。这样的话,这个数的pos在哪,他应该对应插入在第pos个空格的位置。注意本题序号从0开始,要加一。
这样就豁然开朗啦!!真是神奇的操作!
看一下代码。
#include<iostream> #include<cstdio> #include<cmath> #include<algorithm> #include<queue> #include<cstring> #include<utility> #include<map> #define pr pair<int,int> #define mp make_pair #define fi first #define sc second #define rep(i,a,n) for(int i = a;i <= n;i++) #define per(i,n,a) for(int i = n;i >= a;i--) #define enter putchar(‘\n‘) using namespace std; typedef long long ll; const int M = 200005; const int N = 10000005; int read() { int ans = 0,op = 1; char ch = getchar(); while(ch < ‘0‘ || ch > ‘9‘) { if(ch == ‘-‘) op = -1; ch = getchar(); } while(ch >=‘0‘ && ch <= ‘9‘) { ans *= 10; ans += ch - ‘0‘; ch = getchar(); } return ans * op; } struct node { int id,val; }q[M]; int n,sum[M<<2],a[M]; void clear() { memset(q,0,sizeof(q)); memset(sum,0,sizeof(sum)); } void build(int p,int l,int r) { if(l == r) { sum[p] = 1; return; } int mid = (l+r) >> 1; build(p<<1,l,mid),build(p<<1|1,mid+1,r); sum[p] = sum[p<<1] + sum[p<<1|1]; } void modify(int p,int l,int r,int pos,int val) { if(l == r) { a[l] = val,sum[p]--; return; } int mid = (l+r) >> 1; if(sum[p<<1] >= pos) modify(p<<1,l,mid,pos,val); else modify(p<<1|1,mid+1,r,pos-sum[p<<1],val); sum[p] = sum[p<<1] + sum[p<<1|1]; } int main() { while(scanf("%d",&n) != EOF) { build(1,1,n); rep(i,1,n) q[i].id = read() + 1,q[i].val = read(); per(i,n,1) modify(1,1,n,q[i].id,q[i].val); rep(i,1,n) printf("%d ",a[i]);enter; } return 0; }
标签:getch || nod 代码 main i++ return etc stream
原文地址:https://www.cnblogs.com/captain1/p/9795207.html