zoukankan      html  css  js  c++  java
  • 51Nod 1753 相似子串

    题目大意:
    两个字符串相似定义为:
    1.两个字符串长度相等
    2.两个字符串对应位置上有且仅有至多一个位置所对应的字符不相同
    给定一个字符串,每次询问两个子串在给定的规则下是否相似。给定的规则指每次给出一些等价关系,如‘a'=’b',‘b'=’c'等,注意这里的等价关系具有传递性,即若‘a'=’b',‘b'=’c',则‘a'=’c'。

    解题报告:
    正解:哈希+二分
    我们先处理出每一个字母哈希值的前缀和,然后并查集维护等价关系,对于相似的判断,如果两串的哈希值完全一样,就为YES,如果存在一个字符不一样,那么我们就二分那个位置,判断左右哈希值是否相等,如果两端哈希值都不相等,说明存在两个以上的位置不相同,判为NO

    #include <algorithm>
    #include <iostream>
    #include <cstdlib>
    #include <cstring>
    #include <cstdio>
    #include <cmath>
    #define RG register
    #define il inline
    #define iter iterator
    #define Max(a,b) ((a)>(b)?(a):(b))
    #define Min(a,b) ((a)<(b)?(a):(b))
    using namespace std;
    typedef long long ll;
    const int N=300005;
    ll p=20021021,mod=1e9+7,ha[N][26],mul[N];
    char S[N],sc[5];int n,a[N],fa[27];
    int find(int x){return fa[x]==x?x:fa[x]=find(fa[x]);}
    ll query(int l,int r){
    	ll ret=0,x;
    	for(int i=0;i<26;i++){
    		x=((ha[r][i]-(ha[l-1][i]*mul[r-l+1])%mod)+mod)%mod;
    		ret=(ret+x*(find(i)+1))%mod;
    	}
    	return ret;
    }
    void work()
    {
    	scanf("%s",S+1);
    	n=strlen(S+1);
    	for(int i=1;i<=n;i++)a[i]=S[i]-'a';
    	mul[0]=1;for(int i=1;i<=n;i++)mul[i]=mul[i-1]*p%mod;
    	for(int i=1;i<=n;i++){
    		for(int j=0;j<26;j++)
    			ha[i][j]=ha[i-1][j]*p%mod;
    		(ha[i][a[i]]+=1)%=mod;
    	}
    	int Q,Ls,Rs,Lt,Rt,m,l,r,mid,Hl,Hr,Hll,Hrr,flag;
    	scanf("%d",&Q);
    	while(Q--){
    		scanf("%d%d%d%d%d",&m,&Ls,&Rs,&Lt,&Rt);
    		for(int i=0;i<26;i++)fa[i]=i;
    		for(int i=1;i<=m;i++){
    			scanf("%s",sc);
    			if(find(sc[0]-'a')!=find(sc[1]-'a'))
    				fa[find(sc[1]-'a')]=find(sc[0]-'a');
    		}
    		if(Rt-Lt!=Rs-Ls){puts("NO");continue;}
    		if(query(Lt,Rt)==query(Ls,Rs)){puts("YES");continue;}
    		l=1;r=Rt-Lt+1;flag=0;
    		while(l<=r){
    			mid=(l+r)>>1;
    			Hl=query(Ls,Ls+mid-1);Hr=query(Lt,Lt+mid-1);
    			Hll=query(Ls+mid,Rs);Hrr=query(Lt+mid,Rt);
    			if(Hl!=Hr && Hll!=Hrr){flag=1;break;}
    			if(Hl!=Hr)r=mid-1;
    			else l=mid+1;
    		}
    		if(flag)puts("NO");
    		else puts("YES");
    	}
    }
    
    int main()
    {
    	work();
    	return 0;
    }
    
    
  • 相关阅读:
    Redis笔记
    java多线程 interrupt(), interrupted(), isInterrupted()方法区别
    策略模式
    外观模式
    Java线程池原理与架构分析
    状态模式
    模板方法模式
    LeetCode | Path-Sum
    实用工具
    IDEA springboot配置
  • 原文地址:https://www.cnblogs.com/Yuzao/p/7565452.html
Copyright © 2011-2022 走看看