不明觉厉的数据结构题
这篇文章会列一点非常规的数据结构题。
tyvj4393 纸条
大意是有若干条纸条,每次操作是把一个纸条挪动一下,然后求一个位置的所有纸条上数的和。所有纸条的长度之和不超过20W,纸条数量啥的都是10W。
首先我们可以分块,不过标解不知道高到哪里去了,对于所有的纸条按照长度大于小于$\sqrt{n}$分个类,小于$\sqrt{n}$的挪动时暴力修改,大于$\sqrt{n}$的反正不到$\sqrt{n}$个,询问时枚举一遍就行。
求老司机讲一下线段树的做法…
#include <iostream> #include <stdio.h> #include <stdlib.h> #include <algorithm> #include <string.h> #include <vector> #include <math.h> #include <limits> #include <set> #include <map> using namespace std; #define SZ 233333 #define MP 450 int n,m,q,lp[SZ],M=1,sn=0,ln=0,fff[SZ],lid[SZ],sid[SZ]; struct line {int st,cnt,pos;}sma[SZ],lar[SZ]; int main() { scanf("%d%d%d",&n,&m,&q); for(int i=1;i<=m;i++) { line cur; cur.st=M; scanf("%d%d",&cur.pos,&cur.cnt); for(int j=0;j<cur.cnt;j++) scanf("%d",&lp[M++]); if(cur.cnt>MP) lar[++ln]=cur, lid[i]=ln; else sma[++sn]=cur, sid[i]=sn; } for(int i=1;i<=sn;i++) { for(int j=0;j<sma[i].cnt;j++) fff[sma[i].pos+j]+=lp[sma[i].st+j]; } while(q--) { char ps[6]; int a,b; scanf("%s",ps); if(ps[0]=='Q') { scanf("%d",&a); --a; int ans=fff[a]; for(int j=1;j<=ln;j++) { if(lar[j].pos<=a&&a<=lar[j].pos+lar[j].cnt-1) ans+=lp[lar[j].st+a-lar[j].pos]; } printf("%d\n",ans); } else { scanf("%d%d",&a,&b); if(lid[a]) {lar[lid[a]].pos=b; continue;} int id=sid[a]; for(int j=0;j<sma[id].cnt;j++) { fff[sma[id].pos+j]-=lp[sma[id].st+j]; fff[b+j]+=lp[sma[id].st+j]; } sma[id].pos=b; } } }
tyvj4394 超空间
矩阵加法,询问某一历史版本矩阵的和,回溯到某一历史版本。
矩阵边长在1000以内,询问在2000以内。
可持久化二维树套树?
我们发现这些操作实际上构成了一棵类似操作树的东西。
在这个操作树上我们考虑查询,那么只有操作路径上的东西才会有贡献。
暴力模拟即可。于是就是一道傻逼题。
#include <iostream> #include <stdio.h> #include <stdlib.h> #include <algorithm> #include <string.h> #include <vector> #include <math.h> #include <limits> #include <set> #include <map> using namespace std; typedef long long ll; #define SZ 233333 int ct=0,n,q,x1[SZ],x2[SZ],y1[SZ],y2[SZ],vv[SZ],fa[SZ]; int main() { scanf("%d%d",&n,&q); int cur=0; for(int i=1;i<=q;i++) { char s[7]; scanf("%s",s); if(s[1]=='a'||s[1]=='o') //Raise/Lower { int a,b,c,d,e; scanf("%d%d%d%d%d",&a,&b,&c,&d,&e); e*=(s[1]=='o')?-1:1; ++ct; x1[ct]=a; y1[ct]=b; x2[ct]=c; y2[ct]=d; vv[ct]=e; fa[ct]=cur; cur=ct; } else if(s[1]=='u') //Query { int t,_x1,_y1,_x2,_y2; scanf("%d%d%d%d%d",&t,&_x1,&_y1,&_x2,&_y2); ll ans=0; while(t) { int x_1=max(_x1,x1[t]); int y_1=max(_y1,y1[t]); int x_2=min(_x2,x2[t]); int y_2=min(_y2,y2[t]); ans+=(ll)vv[t]*max(x_2-x_1+1,0)*max(y_2-y_1+1,0); t=fa[t]; } printf("%lld\n",ans); } else { int x; scanf("%d",&x); cur=x; } } }
嗯如果询问10W?
离线地做。
在树上dfs的时候顺便修改一下二维树状数组似乎就行。
如果边长10W?
在树上dfs的时候顺便修改一下二维线段树似乎就行。
感觉又造出了一道看起来很厉害的数据结构题…
矩阵加法,询问某一历史版本矩阵的和,回溯到某一历史版本。
矩阵边长和询问数量在10W以内。
代码不想写了…谁爱写谁写
本来还想放几道题的…但是题解写不出来等有空再补