Hdu 6863

Isomorphic Strings

题目大意:给你一个字符串,问你是否能将这个字符串分为k个相同地字符串,并且这k个字符串同构。

同构:如果将一个字符串的前几个字符按顺序移到后面,使得这个字符串和另一个字符串相同,那么这两个字符串同构

解题思路:hash来求解,求长度的每一个因子,然后可以将这个串分为若干个长度相同地字串。然后我们求得第一个字串的所有同构字串的hash值,判断剩下字串的hash值是否都包含在这个集合里即可

Hash:https://blog.csdn.net/nka_kun/article/details/81254013

#include <bits/stdc++.h>
using namespace std;
typedef long long ll;
const int mod=1e7+7;
const int base=1331;
const int maxn=5e6+10;
int t,n;
char s[maxn];
int  p[maxn],a[maxn];
int jl[mod];
inline ll read(){
    ll s=0,w=1;char ch = getchar();
    while(ch<48 || ch>57) {
        if(ch=='-') w=-1;ch = getchar();
    }
    while(ch>=48&&ch<=57) s = (s<<1) + (s<<3) + (ch^48),ch=getchar();
    return s*w;
}
int js(int l,int r)
{
    return (a[r]-1ll*a[l-1]*p[r-l+1]%mod+mod)%mod;
}
int tot=0;
bool pd(int k,int id)
{
    if(k==1) return false;
    int len=n/k;
    int l=1,r=len;
    jl[js(1,len)]=id;
    for(int i=1;i<len;i++)
    {
        int temp1=(1ll*js(i+1,len)*p[i]%mod+js(1,i))%mod;
        jl[temp1]=id;
        
    }
    for(int i=1;i<=k;i++)
    {
        int temp=js(l,r);
        if(jl[temp]!=id) return  false;
        l+=len,r+=len;
    }
    return true;
}
int main()
{
    int t;
    t=read();
    p[0]=1;
    for(int  i=1;i<maxn;i++) p[i]=(1ll*p[i-1]*base)%mod;
    while(t--)
    {
        n=read();
        scanf("%s",s+1);
        for(int i=1;i<=n;i++) a[i]=(1ll*a[i-1]*base%mod+(s[i]-'a'+1))%mod;
        int flag=0;
        for(int i=1;1ll*i*i<=1ll*n&&!flag;i++)
        {
            if(n%i==0)
            {
                flag|=pd(i,tot++);
                flag|=pd(n/i,tot++);
            }
        }
        if(!flag) cout<<"No"<<endl;
        else cout<<"Yes"<<endl;
    }
    return 0;
}

  

posted @ 2020-08-14 17:17  mcalex  阅读(146)  评论(0编辑  收藏  举报