#C241118B. 宠物狗
标签(Label)
- 哈希
- 集合哈希
网址(Website)
题目(Problem)
样例
输入数据 1
2
aab
ababccd
输出数据 1
6
661
【样例 1 解释】
对于第一组询问:
- $d=1$ 时分割方式为 ${[1,1],[2,2],[3,3]}$,有 $3$ 种不同的方案;
- $d=2$ 时分割方式为 ${[1,2],[3,3]},{[1,1],[2,3]}$,其中 $[1,1],[3,3]$ 长度不为 $d$,不用于排列,有 $2$ 种不同的方案;
- $d=3$ 时分割方式为 ${[1,3]}$,有 $1$ 种不同的方案。共 $6$ 种方案。
对于第二组询问,共有 $661$ 种不同的方案。
【样例 2】
见选手目录下的 dog/dog2.in 和 dog/dog2.ans。
该样例满足子任务 $1$ 的限制。
【样例 3】
见选手目录下的 dog/dog3.in 和 dog/dog3.ans。
该样例满足子任务 $2$ 的限制。
【样例 4】
见选手目录下的 dog/dog4.in 和 dog/dog4.ans。
该样例满足子任务 $3$ 的限制。
【样例 5】
见选手目录下的 dog/dog5.in 和 dog/dog5.ans。
该样例满足子任务 $4$ 的限制。
【样例 6】
见选手目录下的 dog/dog6.in 和 dog/dog6.ans。
该样例满足子任务 $5$ 的限制。
【样例 7】
见选手目录下的 dog/dog7.in 和 dog/dog7.ans。
该样例满足子任务 $6$ 的限制。
题解(Solution)
很明显发现枚举区间长度 $d$ 之后获得每个区间的时间是调和级数的时间复杂度 $O(n\ln n)$ ,而每次枚举剩下的区间也是 $O(n\ln n)$ 的。后面就考虑使用 $\text{Hash}$ 以及 $\verb!map!$ 来计算区间数量,如果区间互不相同,那么最后的答案就是 $\lfloor\frac{n}{d}\rfloor!$ ,如果有相同的区间,那么令相同的区间数量分别为 $x_1,x_2,x_3,\cdots,x_s$ ,最后的答案就是 $\frac{\lfloor\frac{n}{d}\rfloor!}{\Pi_{i=1}^{s}x_i}$ 。
但是这个时候发现,因为每次只会删除一部分区间,可能出现情况相同导致重复计算的情况,这个地方相当于是要判断集合是否相等,每个区间就是集合里面的一个元素,考虑使用手写表,给每个区间对应一个随机的权值,分别维护这些区间的异或和和加和,存到 $\texttt{map}$ 里面,这样就能判断集合是否相等了。
但是由于 $\texttt{map}$ 自带 $O(\log n)$ 的常数,我们可以使用 $\texttt{unordered_map}$ 来存储,可是常数还是很大,无奈只能手写哈希表,大概开 $800$ 个 $\texttt{vector}$ ,每次加入一个点将其对 $800$ 取模,然后存到对应的空间里面,加入新点的过程相当于建边,查询就在对应的地方暴力查询就好了。由于一共最多加入 $n\ln n$ 次,分成 $800$ 份之后能省下很多常数,实现的时候可以看本地测试判断开多少 $\texttt{vector}$ 更优。使用链式前向星更快一些,但是都能过。
时间复杂度:$O(n\ln n)$ 。
出题者题解
代码(Code)
75分(使用map)
#include<bits/stdc++.h> #define For(i,l,r) for(int i=l;i<=r;i++) #define Rof(i,l,r) for(int i=l;i>=r;i--) using namespace std; #define P pair<int,int> #define int long long #define x first #define y second inline int rd(){ char c;bool f=false;while(!isdigit(c=getchar()))f=c=='-';int x=c^48; while(isdigit(c=getchar())){x=(((x<<2)+x)<<1)+(c^48);}return f?-x:x; } inline void write(int x){ if(x<0) putchar('-'), write(-x); else if(x<10) putchar(x+'0'); else write(x/10), putchar(x%10+'0'); }bool ST; const int inf = 0x3f3f3f3f3f3f3f3f; const int N = 212345; const int mod = 998244353; mt19937_64 rr(time(0)); inline int rnd(int l,int r){return rr()%(r-l+1)+l;} int fac[N],inv[N]; inline void add(int &x,int y){if((x+=y)>=mod)x-=mod;} inline void mul(int &x,int y){x=x*y%mod;} inline int ksm(int x,int k=mod-2){ int res=1;while(k){ if(k&1) mul(res,x); mul(x,x), k>>=1; }return res; }inline int C(int n,int m){return fac[n]*inv[m]%mod*inv[n-m]%mod;} char s[N]; int n,ans; namespace H{ const int p1 = 1e9+9, b1 = 1331; const int p2 = 1e9+7, b2 = 997; int fac1[N], pw1[N]; int fac2[N], pw2[N]; void init(){ pw1[0]=1;For(i,1,N-5) pw1[i]=pw1[i-1]*b1%p1; pw2[0]=1;For(i,1,N-5) pw2[i]=pw2[i-1]*b2%p2; } void pre(){ For(i,1,n) fac1[i] = (fac1[i-1]*b1 + s[i]-'a'+1)%p1; For(i,1,n) fac2[i] = (fac2[i-1]*b2 + s[i]-'a'+1)%p2; } inline int calc(int l,int r){ int t1 = (fac1[r]-pw1[r-l+1]*fac1[l-1]%p1+p1)%p1; int t2 = (fac2[r]-pw2[r-l+1]*fac2[l-1]%p2+p2)%p2; return (t1*131ll+t2); } } map<int,int> mp,hsh; map<P,int> exi; int Val, S1, S2; inline void ins(int x){ mul(Val, fac[mp[x]]); mp[x]++, mul(Val, inv[mp[x]]); if(!hsh.count(x)) hsh[x] = rnd(1,1e9); S1 += hsh[x], S2 ^= hsh[x]; } inline void del(int x){ mul(Val, fac[mp[x]]), mp[x]--; mul(Val, inv[mp[x]]); assert(hsh.count(x)); S1 -= hsh[x], S2 ^= hsh[x]; } void Solve(){ ans = 0; scanf("%s", s+1), n = strlen(s+1); H::pre(); For(d,1,n){ mp.clear(), exi.clear(), hsh.clear(); Val = fac[n/d], S1 = 0, S2 = 0; int yu = n%d; for(int i=yu+1;i<=n;i+=d){//假定n%d起始点在1,存入 hash ins(H::calc(i,i+d-1)); } add(ans, Val); if(n%d==0) continue; exi[{S1,S2}] = true; for(int i=1,j=yu+1;j<=n;i+=d,j+=d){//枚举n%d的起始点 ins(H::calc(i,i+d-1)), del(H::calc(j,j+d-1)); if(!exi.count({S1,S2})){ exi[{S1,S2}] = true; add(ans, Val); } } }write(ans),puts(""); } bool ED; //多测清空 signed main(){ cerr<<abs(&ST-&ED)/1024./1024.<<"MB\n"; freopen("dog.in","r",stdin); freopen("dog.out","w",stdout); int Tt=rd();double Tim=clock(); fac[0]=1;For(i,1,N-5) fac[i]=fac[i-1]*i%mod; inv[N-5] = ksm(fac[N-5]); Rof(i,N-6,0) inv[i] = inv[i+1]*(i+1)%mod; H::init();while(Tt--) Solve(); return cerr<<"TIME:"<<(clock()-Tim)/CLOCKS_PER_SEC,0; }
100分(使用邻接表)
#include<bits/stdc++.h> #define For(i,l,r) for(int i=l;i<=r;i++) #define Rof(i,l,r) for(int i=l;i>=r;i--) using namespace std; #define P pair<int,int> #define int long long #define x first #define y second inline int rd(){ char c;bool f=false;while(!isdigit(c=getchar()))f=c=='-';int x=c^48; while(isdigit(c=getchar())){x=(((x<<2)+x)<<1)+(c^48);}return f?-x:x; } inline void write(int x){ if(x<0) putchar('-'), write(-x); else if(x<10) putchar(x+'0'); else write(x/10), putchar(x%10+'0'); }bool ST; const int inf = 0x3f3f3f3f3f3f3f3f; const int N = 201234; const int mod = 998244353; mt19937_64 rr(time(0)); inline int rnd(int l,int r){return rr()%(r-l+1)+l;} int fac[N],inv[N]; inline void add(int &x,int y){if((x+=y)>=mod)x-=mod;} inline void mul(int &x,int y){x=x*y%mod;} inline int ksm(int x,int k=mod-2){ int res=1;while(k){ if(k&1) mul(res,x); mul(x,x), k>>=1; }return res; } char s[N]; int n,ans; namespace H{ const int p1 = 1e9+9, b1 = 131; const int p2 = 1e9+7, b2 = 233; int fac1[N], pw1[N]; int fac2[N], pw2[N]; void init(){ pw1[0]=1;For(i,1,N-5) pw1[i]=pw1[i-1]*b1%p1; pw2[0]=1;For(i,1,N-5) pw2[i]=pw2[i-1]*b2%p2; } void pre(){ For(i,1,n) fac1[i] = (fac1[i-1]*b1 + s[i]-'a'+1)%p1; For(i,1,n) fac2[i] = (fac2[i-1]*b2 + s[i]-'a'+1)%p2; } inline int calc(int l,int r){ int t1 = (fac1[r]-pw1[r-l+1]*fac1[l-1]%p1+p1)%p1; int t2 = (fac2[r]-pw2[r-l+1]*fac2[l-1]%p2+p2)%p2; return (t1<<31)+t2; } } const int M = 800; struct HL{ int hd[805], nxt[N<<3|7], tot; int num[N<<3|7], sum[N<<3|7]; inline int add(int x,int v=1){ int tmp = x%M; for(int i=hd[tmp];i;i=nxt[i]) if(num[i]==x) return sum[i]+=v; nxt[++tot]=hd[tmp], hd[tmp] = tot; num[tot] = x, sum[tot] = v;return v; } inline int chkins(int x,int v){ int tmp = x%M; for(int i=hd[tmp];i;i=nxt[i]) if(num[i]==x) return sum[i]; nxt[++tot]=hd[tmp], hd[tmp] = tot; num[tot] = x, sum[tot] = v;return v; } inline int operator[](int x){ int tmp = x%M; for(int i=hd[tmp];i;i=nxt[i]) if(num[i]==x) return sum[i]; return 0; } inline void clear(){memset(hd,0,sizeof hd), tot=0;} }mp,hsh,exi; int Val, S1, S2; inline void ins(int x){ int tmp = mp.add(x,1); mul(Val, fac[tmp-1]); mul(Val, inv[tmp]); tmp = hsh.chkins(x,rnd(1,1e12)), S1 += tmp, S2 ^= tmp; } inline void del(int x){ int tmp = mp.add(x,-1); mul(Val, fac[tmp+1]); mul(Val, inv[tmp]); tmp = hsh.chkins(x,rnd(1,1e12)), S1 -= tmp, S2 ^= tmp; } inline int zhuan(int x,int y){return x*37+y;} void Solve(){ ans = 0; scanf("%s", s+1), n = strlen(s+1); H::pre(); For(d,1,n){ mp.clear(), exi.clear(), hsh.clear(); Val = fac[n/d], S1 = 0, S2 = 0; int yu = n%d; for(int i=yu+1;i<=n;i+=d){ ins(H::calc(i,i+d-1)); } add(ans, Val); if(n%d==0) continue; exi.add(zhuan(S1,S2)); for(int i=1,j=yu+1;j<=n;i+=d,j+=d){ ins(H::calc(i,i+d-1)), del(H::calc(j,j+d-1)); if(exi.add(zhuan(S1,S2)) == 1){ add(ans, Val); } } }write(ans),puts(""); } bool ED; signed main(){ cerr<<abs(&ST-&ED)/1024./1024.<<"MB\n"; freopen("dog.in","r",stdin); freopen("dog.out","w",stdout); int Tt=rd();double Tim=clock(); fac[0]=1;For(i,1,N-5) fac[i]=fac[i-1]*i%mod; inv[N-5] = ksm(fac[N-5]); Rof(i,N-6,0) inv[i] = inv[i+1]*(i+1)%mod; H::init();while(Tt--) Solve(); return cerr<<"TIME:"<<(clock()-Tim)/CLOCKS_PER_SEC,0; }
100分(使用vector)
#include<bits/stdc++.h> #define For(i,l,r) for(int i=l;i<=r;i++) #define Rof(i,l,r) for(int i=l;i>=r;i--) using namespace std; #define P pair<int,int> #define int long long #define x first #define y second inline int rd(){ char c;bool f=false;while(!isdigit(c=getchar()))f=c=='-';int x=c^48; while(isdigit(c=getchar())){x=(((x<<2)+x)<<1)+(c^48);}return f?-x:x; } inline void write(int x){ if(x<0) putchar('-'), write(-x); else if(x<10) putchar(x+'0'); else write(x/10), putchar(x%10+'0'); }bool ST; const int inf = 0x3f3f3f3f3f3f3f3f; const int N = 1012345; const int mod = 998244353; mt19937_64 rr(time(0)); inline int rnd(int l,int r){return rr()%(r-l+1)+l;} int fac[N],inv[N]; inline void add(int &x,int y){if((x+=y)>=mod)x-=mod;} inline void mul(int &x,int y){x=x*y%mod;} inline int ksm(int x,int k=mod-2){ int res=1;while(k){ if(k&1) mul(res,x); mul(x,x), k>>=1; }return res; } char s[N]; int n,ans; namespace H{ const int p1 = 1e9-63, b1 = 131; const int p2 = 1e9+7, b2 = 997; int fac1[N], pw1[N]; int fac2[N], pw2[N]; void init(){ pw1[0]=1;For(i,1,N-5) pw1[i]=pw1[i-1]*b1%p1; pw2[0]=1;For(i,1,N-5) pw2[i]=pw2[i-1]*b2%p2; } void pre(){ For(i,1,n) fac1[i] = (fac1[i-1]*b1 + s[i]-'a'+1)%p1; For(i,1,n) fac2[i] = (fac2[i-1]*b2 + s[i]-'a'+1)%p2; } inline int calc(int l,int r){ int t1 = (fac1[r]-pw1[r-l+1]*fac1[l-1]%p1+p1)%p1; int t2 = (fac2[r]-pw2[r-l+1]*fac2[l-1]%p2+p2)%p2; return (t1*37+t2); } } const int M = 200; struct Hash_List{ vector<P> link[1024]; inline int add(int x,int v=1){ for(auto &p:link[x%M]) if(p.x==x) return p.y+=v; link[x%M].emplace_back(x,v);return v; } inline int chkins(int x,int v){ for(auto &p:link[x%M]) if(p.x==x) return p.y; link[x%M].emplace_back(x,v);return v; } inline int operator[](int x){ for(auto &p:link[x%M]) if(x==p.x) return p.y; return 0; } void clear(){For(i,0,M-1) link[i].clear();} }mp,hsh,exi; int Val, S1, S2; inline void ins(int x){ int tmp = mp.add(x,1); mul(Val, fac[tmp-1]); mul(Val, inv[tmp]); tmp = hsh.chkins(x,rnd(1,1e12)), S1 += tmp, S2 ^= tmp; } inline void del(int x){ int tmp = mp.add(x,-1); mul(Val, fac[tmp+1]); mul(Val, inv[tmp]); tmp = hsh.chkins(x,rnd(1,1e12)), S1 -= tmp, S2 ^= tmp; } inline int zhuan(int x,int y){return x*37+y;} void Solve(){ ans = 0; scanf("%s", s+1), n = strlen(s+1); H::pre(); For(d,1,n){ mp.clear(), exi.clear(), hsh.clear(); Val = fac[n/d], S1 = 0, S2 = 0; int yu = n%d; for(int i=yu+1;i<=n;i+=d){//假定n%d起始点在1,存入 hash ins(H::calc(i,i+d-1)); } add(ans, Val); if(n%d==0) continue; exi.add(zhuan(S1,S2)); for(int i=1,j=yu+1;j<=n;i+=d,j+=d){//枚举n%d的起始点 ins(H::calc(i,i+d-1)), del(H::calc(j,j+d-1)); if(exi.add(zhuan(S1,S2)) == 1){ add(ans, Val); } } }write(ans),puts(""); } bool ED; signed main(){ cerr<<abs(&ST-&ED)/1024./1024.<<"MB\n"; freopen("dog.in","r",stdin); freopen("dog.out","w",stdout); int Tt=rd();double Tim=clock(); fac[0]=1;For(i,1,N-5) fac[i]=fac[i-1]*i%mod; inv[N-5] = ksm(fac[N-5]); Rof(i,N-6,0) inv[i] = inv[i+1]*(i+1)%mod; H::init();while(Tt--) Solve(); return cerr<<"TIME:"<<(clock()-Tim)/CLOCKS_PER_SEC,0; }