zoukankan      html  css  js  c++  java
  • [考试反思]0306省选模拟39:延续

     

    俩原题。对于看视频课太匆忙的我来说,还是算俩新题吧。

     为啥感觉最近总是考第$9$啊。最近这种名次的频率好高。。

    (好菜啊第$9$还学个啥呢啊脑子呢???这还翻个什么盘啊)

    然而其实第$9$也是混来的。

    最开始先看的$T1$发现好像是原题,但是印象不深,大概还记得第一步怎么做。

    接着往后看,$T2$看起来像弱智题。$T3$的话又是原题,思路极其简单但是是$exSAM+LCT$写出来会暴毙。于是扔在最后。

    然后回$T1$尝试回忆。然后就没有然后了。折腾来折腾去想了两个小时硬是没想到$dp$定义。

    果断换题。然后一看$T2$,好像的确有点弱智啊。

    思路大概是对的,但是在枚举三元环上卡了一下。然后就真的忘记了怎么$O(m sqrt{m})$枚举三元环了。

    然后的确就一直没有想起来。又耗了一个小时,最后利用大小点卡常数,理论上时间复杂度还是$O(n^2)$

    但是实际上常数非常小,我自己貌似也构造出什么数据卡满我自己。然而时间也不多了,调着调着就交了。

    最后剩了十几分钟给$T3$。尝试爆写$exSAM$最后还是放弃了。

    我为啥又把暴跳父亲忘了???如果想起来的话这次就能有两道题都是$n^2$过$2 imes 10^5$了!

    然而既然考场上没写。。。那考后就老老实实写正解吧。。。

    强烈谴责RNB考后暴力暴跳父亲AC的无脸行为

    话说我自己数据结构能力是真的弱啊

    T1:gift

    大意:两数列$a,b$。有些位置为空,问有多少种方案填数使之成为排列,满足通过至少$k$次交换$a$变成$b$。对所有$k in [0,n-1]$求解。$n le 2000$

    如果序列已经填满了数,那么最少交换次数可以这么表示:所有$a_i ightarrow b_i$连边。然后每个环需要交换$size-1$次。所以交换次数就是$n-$环个数。

    现在有些位置为$0$了,那么所有的边分四类$0-x,x-y,x-0,0-0$。而有些边已经有公共点了($0$除外)。把它们缩成一条链

    对于$x-y$这一类,我们发现在缩成链之后两端点都未消失,证明它们都只出现了一次。

    我们可以把它们两个当成同一个数看待,可以发现这对最后的图没有影响,在最后的时候展开就行了。

    所以边只剩下三类,我们分别讨论它们之间的关系:

    我们发现,$0-x$这种边它的$x$一定只出现了一次,所以一定与$0$相连了,那么要么就是接上了一个$0-0$,要么就是接上了另一个$0-x$,或者说自己连上自己变成自环。

    对于第一种情况,新的两个端点就是$0-0$了,那么相当于这条边变成$0-0$了,而同时$0-0$的总量加一减一后并没有改变。

    对于第二种情况,那么就是合并成了一个新的$0-x$。对最后的环数答案并没有什么影响。

    所以我们只在意$0-x$这一类边自己形成了多少个环,以及方案数。

    直接算不好算,考虑容斥,得到$i$个环的方案数是$g_i =sumlimits_{j=i}^{cnt_{0-x}} inom{cnt_{0-x}}{j} egin{bmatrix} j \ i end{bmatrix} (cnt_{0-0}+cnt_{0-x}-j)^{underline{cnt_{0-x}-j}}$

    意思就是说,枚举有多少条$0-x$边在环里,对于剩下的边,我们在$cnt_{0-0}$个$0-0$边和$cnt_{0-x}$个$0-x$边中任意选择一条接在后面。方案数是下降幂。

    后面乱选的部分也有可能形成环,所以这个$g$是至少的形式。用一个简单的二项式反演就能得到恰好的形式。

    对于所有的$x-0$边,同理。它和$0-x$边如果要产生关系,一定要借助$0-0$而上面的第一种情况说了只要用到$0-0$就会变成$0-0$所以只需要考虑$x-0$与自身以及$0-0$的关系。

    这两部分像个背包。直接$OGF$卷起来就好了。得到的是把所有的$0-x$边以及$x-0$边连成环或者变味$0-0$的方案数了。

    所以现在只需要继续考虑$cnt_{0-0}$条$0-0$边就好了。

    这个的方案数比较好算,环的形状一共是第二类斯特林数种,然后你再给它们赋予$cnt_{0,0}$的先后顺序,也就是个阶乘。

    一起卷起来就得到答案了。

    $Dy$讲的时候真实啥也没听懂,自己回放的时候其实也不明白,说啥都没有自己做一遍有效。

     1 #include<bits/stdc++.h>
     2 using namespace std;
     3 #define mod 998244353
     4 int mo(int a){return a>=mod?a-mod:a;}
     5 #define S 2002
     6 int n,a[S],b[S],al[S],c00,c01,c10,cir,in[S],out[S],C[S][S],s[S][S],A[S][S],G[S],g[S],H[S],h[S],f[S],ans[S];
     7 int main(){
     8     scanf("%d",&n);
     9     for(int i=1;i<=n;++i)scanf("%d",&a[i]);
    10     for(int i=1;i<=n;++i)scanf("%d",&b[i]);
    11     for(int i=1;i<=n;++i)in[i]=out[i]=-1;
    12     for(int i=1;i<=n;++i)out[a[i]]=b[i],in[b[i]]=a[i];
    13     for(int i=1;i<=n;++i)if(in[i]==-1&&out[i]!=-1){
    14         int p=i;al[p]=1;
    15         while(p&&out[p]!=-1)p=out[p],al[p]=1;
    16         if(!p)c10++;
    17     }
    18     for(int i=1;i<=n;++i)if(in[i]==0){
    19         int p=i;al[p]=1;
    20         while(p&&out[p]!=-1)p=out[p],al[p]=1;
    21         if(!p)c00++;else c01++;
    22     }
    23     for(int i=1;i<=n;++i)c00+=a[i]==0&&b[i]==0;
    24     for(int i=1;i<=n;++i)if(in[i]>0&&out[i]>0&&!al[i]){
    25         int p=i;cir++;
    26         while(!al[p])al[p]=1,p=out[p];
    27     }
    28     for(int i=0;i<=n;++i)for(int j=C[i][0]=1;j<=i;++j)C[i][j]=mo(C[i-1][j-1]+C[i-1][j]);
    29     for(int i=s[0][0]=1;i<=n;++i)for(int j=1;j<=i;++j)s[i][j]=(s[i-1][j-1]+s[i-1][j]*(i-1ll))%mod;
    30     for(int i=0;i<=n;++i)for(int j=A[i][0]=1;j<=i;++j)A[i][j]=A[i][j-1]*(i+1ll-j)%mod;
    31     for(int i=0;i<=c01;++i)for(int j=i;j<=c01;++j)g[i]=(g[i]+1ll*C[c01][j]*s[j][i]%mod*A[c01+c00-j][c01-j])%mod;
    32     for(int i=0;i<=c01;++i)for(int j=i;j<=c01;++j)G[i]=(G[i]+(j-i&1?mod-1ll:1ll)*g[j]%mod*C[j][i])%mod;
    33     for(int i=0;i<=c10;++i)for(int j=i;j<=c10;++j)h[i]=(h[i]+1ll*C[c10][j]*s[j][i]%mod*A[c10+c00-j][c10-j])%mod;
    34     for(int i=0;i<=c10;++i)for(int j=i;j<=c10;++j)H[i]=(H[i]+(j-i&1?mod-1ll:1ll)*h[j]%mod*C[j][i])%mod;
    35     for(int i=0;i<=c01;++i)for(int j=0;j<=c10;++j)f[i+j]=(f[i+j]+1ll*G[i]*H[j])%mod;
    36     for(int i=0;i<=c01+c10;++i)for(int j=0;j<=c00;++j)ans[i+j]=(ans[i+j]+1ll*f[i]*s[c00][j]%mod*A[c00][c00])%mod;
    37     for(int i=0;i<n;++i)printf("%d ",n>=i+cir?ans[n-i-cir]:0);
    38 }
    View Code

    T2:girls

    大意:$[0,n)$。定义合法三元组为三者之间均无连边且$i<j<k$。贡献为$Ai+Bj+Ck$。求总贡献。$n,m le 2 imes 10^5$

    这题也就是经典的容斥了,全部情况加起来减去至少有一条边加上至少有两条边再去掉三元环

    别的都好说,三元环的问题大概就是按照度数排序那一套复杂度$O(m^{1.5})$

    做法也就是把所有的点按照度数排序之后,度数小的向度数大的连边。

    对于点$i$枚举所有出边打标记,再枚举一次出边$j$,再枚举$j$的儿子$k$。若$k$有标记则是三元环。

    联赛前做过,现在忘掉了,每个三元环只在度数最小的点初被枚举,具体复杂度证明忘记了,感性理解一下吧。

    考场上思维僵化只会严格按照$sqrt{n}$为阈值分大小点了,然而一小点两大点的情况不能很好地统计,于是又退化成$O(n^2)$了。

    但是不难发现这样做的常数很小很小。所以$O(n^2)$过$200000$没什么大问题。

     1 #include<bits/stdc++.h>
     2 using namespace std;
     3 #define S 200005
     4 #define ull unsigned long long
     5 ull A,B,C,ans,tot[S],deg[S]; int n,m;
     6 vector<int>v[S],in[S],vs[S],vb[S],inb[S],Bi;
     7 unordered_map<int,bool>M[S];
     8 ull cal(int x){return x*(x-1ull)>>1;}
     9 int main(){
    10     cin>>n>>m>>A>>B>>C; int sq=sqrt(m); 
    11     for(int i=0,a,b;i<m;++i){
    12         scanf("%d%d",&a,&b);
    13         if(a>b)swap(a,b); deg[a]++; deg[b]++; M[a][b]=M[b][a]=1;
    14         v[a].push_back(b); in[b].push_back(a); tot[a]+=b;
    15         ans-=cal(a)*A+a*(a*B+b*C) + (cal(b)-cal(a+1))*B+(b-a-1)*(a*A+b*C) + (cal(n)-cal(b+1))*C+(n-b-1)*(a*A+b*B);
    16     }
    17     for(int i=0;i<n;++i)sort(v[i].begin(),v[i].end()),sort(in[i].begin(),in[i].end());
    18     for(int i=0;i<n;++i)ans+=i*(cal(n-1-i)*A+B*i*(n-1-i)+C*cal(i));
    19     for(int i=0;i<n;++i){
    20         ull x=0,cnt=0;
    21         for(auto j:v[i])ans+=i*cnt*A+x*B+j*cnt*C,x+=j,cnt++;
    22     }
    23     for(int i=0;i<n;++i){
    24         ull x=0,cnt=0;
    25         for(auto j:in[i])ans+=i*cnt*C+x*A+j*cnt*B,x+=j,cnt++;
    26     }
    27     for(int i=0;i<n;++i)for(auto j:v[i])ans+=v[j].size()*(i*A+j*B)+tot[j]*C;
    28     
    29     for(int i=0;i<n;++i)for(auto j:v[i])if(deg[j]>=sq)vb[i].push_back(j);else vs[i].push_back(j);
    30     for(int i=0;i<n;++i)for(auto j:in[i])if(deg[j]>=sq)inb[i].push_back(j);
    31     for(int i=0;i<n;++i)if(deg[i]>=sq&&vb[i].size()>1)for(auto j:vb[i])for(auto k:vb[j])if(M[i][j]&&M[j][k]&&M[i][k])ans-=i*A+j*B+k*C;
    32     for(int i=0;i<n;++i)if(deg[i]<sq&&vs[i].size()>1)for(auto j:vs[i])for(auto k:vs[j])if(M[i][j]&&M[j][k]&&M[i][k])ans-=i*A+j*B+k*C;
    33     for(int i=0;i<n;++i)if(deg[i]<sq&&vs[i].size()>1)for(auto j:vs[i]){
    34         for(auto k:vb[i])if(M[i][j]&&M[j][k]&&M[i][k])ans-=i*A+min(j,k)*B+max(j,k)*C;
    35         for(auto k:inb[i])if(M[i][j]&&M[j][k]&&M[i][k])ans-=k*A+i*B+j*C;
    36     }
    37     
    38     for(int i=0;i<n;++i)if(deg[i]<sq){
    39         for(auto j:inb[i])Bi.push_back(j);
    40         for(auto j:vb[i])Bi.push_back(j);
    41         for(int j=0;j<Bi.size();++j)for(int k=j+1;k<Bi.size();++k){
    42             ull x=i,y=Bi[j],z=Bi[k];
    43             if(x>z)swap(x,z);
    44             if(x>y)swap(x,y);
    45             if(M[x][y]&&M[x][z]&&M[y][z])ans-=A*x+B*y+C*z;
    46         }Bi.clear();
    47     }
    48     cout<<ans<<endl;
    49 }
    View Code

    T3:string

    原题链接。题号12.

    思路很简单,代码很难写。

    干就是了。

     1 #include<bits/stdc++.h>
     2 using namespace std;
     3 #define S 888888
     4 int n,rt=1,pc=1,lst[22],c[11][S],len[S],ans,m,la,f[S],t,w[22][S],lz[22][S];char s[20000003];
     5 vector<int>v[22],P[22];
     6 struct LCT{
     7     int c[2][S],f[S],s[S];
     8     void down(int p){for(int i=1;i<=n;++i)if(lz[i][p])w[i][c[0][p]]+=lz[i][p],w[i][c[1][p]]+=lz[i][p],lz[i][c[0][p]]+=lz[i][p],lz[i][c[1][p]]+=lz[i][p],lz[i][p]=0;} 
     9     bool nr(int p){return c[0][f[p]]==p||c[1][f[p]]==p;}
    10     void spin(int p){
    11         int F=f[p],G=f[F],d=c[1][F]==p,B=c[!d][p];
    12         if(nr(F))c[c[1][G]==F][G]=p; c[!d][p]=F; c[d][F]=B;
    13         f[f[f[B]=F]=p]=G;
    14     }
    15     void splay(int p){
    16         int F=p,t=0; while(nr(F))s[++t]=F,F=f[F]; s[++t]=F; while(t)down(s[t--]);
    17         for(;F=f[p],nr(p);spin(p))if(nr(F))spin(c[1][f[F]]==F^c[1][F]==p?p:F);
    18     }
    19     void access(int p){for(int r=0,s=p;s;s=f[r=s])splay(s),c[1][s]=r;splay(p);}
    20     void link(int a,int b){splay(a);f[a]=b;}
    21     void cut(int p){access(p);f[c[0][p]]=0;c[0][p]=0;}
    22 }T;
    23 void add(int p,int op){T.access(p);for(int i=1;i<=n;++i)lz[i][p]+=op*w[i][p];}
    24 void Fa(int p,int fa){T.link(p,fa);add(p,1);f[p]=fa;}
    25 void extend(int I,int C){
    26     int p=lst[I],np,q,nq;
    27     if(q=c[C][p]){
    28         if(len[q]==len[p]+1){lst[I]=q;T.access(q);lz[I][q]++;w[I][q]++;return;}
    29         nq=++pc; len[nq]=len[p]+1; for(int i=0;i<10;++i)c[i][nq]=c[i][q];
    30         w[I][nq]++;
    31         Fa(nq,f[q]); add(q,-1); T.cut(q); Fa(q,nq);
    32         for(;c[C][p]==q;p=f[p])c[C][p]=nq; lst[I]=nq;
    33     }else{
    34         lst[I]=np=++pc; len[np]=len[p]+1; w[I][np]=1;
    35         for(;p&&!c[C][p];p=f[p])c[C][p]=np;
    36         if(!p){Fa(np,1);ans+=len[np]-len[f[np]];return;}
    37         if(len[q=c[C][p]]==len[p]+1){Fa(np,q);ans+=len[np]-len[f[np]];return;}
    38         nq=++pc; len[nq]=len[p]+1; for(int i=0;i<10;++i)c[i][nq]=c[i][q];
    39         Fa(nq,f[q]); Fa(np,nq); add(q,-1); T.cut(q); Fa(q,nq);
    40         for(;c[C][p]==q;p=f[p])c[C][p]=nq; ans+=len[np]-len[f[np]];
    41     }
    42 }
    43 int main(){//freopen("string1.in","r",stdin);
    44     scanf("%d%d",&n,&t);
    45     for(int i=1;i<=n;++i){
    46         scanf("%s",s);lst[i]=1;
    47         for(int j=0;s[j];++j)extend(i,s[j]-48);
    48         v[i].push_back(0);P[i].push_back(lst[i]);
    49     }scanf("%d",&m);
    50     for(int i=1;i<=m;++i){
    51         int op,x,y,z;scanf("%d",&op);
    52         if(op==1){
    53             scanf("%d%d",&x,&y);
    54             if(t)y^=la,y%=10;
    55             extend(x,y);
    56             v[x].push_back(i);P[x].push_back(lst[x]);
    57         }if(op==2){
    58             scanf("%d%d%d",&x,&y,&z);
    59             int p=P[x][upper_bound(v[x].begin(),v[x].end(),y)-1-v[x].begin()];
    60             T.splay(p);printf("%d
    ",la=w[z][p]);
    61         }if(op==3)printf("%d
    ",ans);
    62         if(op==4){
    63             int p=1;la=0;scanf("%s",s);
    64             for(int j=0;s[j];++j)p=c[s[j]-48][p];
    65             if(p){T.splay(p);for(int j=1;j<=n;++j)la=max(la,w[j][p]);} 
    66             printf("%d
    ",la);
    67         }
    68     }
    69 }
    View Code

    没有调到暴毙还是非常开心的,写的时间比调的时间长。

  • 相关阅读:
    修改oracle用户密码永不过期
    mysql中的union操作(整理)
    mysql条件查询and or使用实例及优先级介绍
    vue实现购物清单列表添加删除
    vue实现全选框效果
    vue实现穿梭框效果
    legend3---13、vue是真的好用
    黑马在线教育项目---34-37、webuploader实现用户头像的异步上传
    lareval重命名created_at和updated_at字段
    js的dom操作(整理)(转)
  • 原文地址:https://www.cnblogs.com/hzoi-DeepinC/p/12431632.html
Copyright © 2011-2022 走看看