zoukankan      html  css  js  c++  java
  • String

    类似Hdu3336

    给出一个字符串,请算出它的每个前缀分别在字符串中出现了多少次。再将这些结果加起来输出
    Input
    The first line include a number T, means the number of test cases.
    For each test case, just a line only include lowercase indicate the String S, the length of S will less than 100000.
    Output
    For each test case, just a number means the sum.
    Sample Input

    3
    acacm
    moreandmorecold
    thisisthisththisisthisisthisththisis

    Sample Output

    7
    19
    82
    HINT
    For the first case,
    there are two "a","ac" and one "aca","acac","acacm" in the string "acacm".
    So the answer is 2 + 2 + 1 + 1 + 1 = 7

    Sol1:利用Kmp的性质,暴力统计,但是会TLE(感觉这种做法是错的)

    #include<bits/stdc++.h>
    using namespace std;
    typedef long long ll;
    ll o,p[300011],sum[300011],len,ans,j;
    char a[300011];
    int main()
    {
    	scanf("%lld",&o);
    	while(o--)
    	{
    		memset(p,0,sizeof(p));
    		memset(sum,0,sizeof(sum));
    		scanf("%s",a+1);
    		len=strlen(a+1);
    		
    		ans=0;j=0;
    		for(ll i=1;i<len;i++)
    		{
    			while(j&&a[j+1]!=a[i+1])
    			    j=p[j];
    			if(a[j+1]==a[i+1])
    			    j++;
    			p[i+1]=j;
    		}
    		for(ll i=1;i<=len;i++)
    		{
    		    sum[i]=1;
    		    ll jj=i;
    		    while (p[jj]!=0)
    		    {
    		    	sum[i]++;
    		    	jj=p[jj];
    		    }
    		    	
    		    ans=ans+sum[i];
    	    }
    		printf("%lld
    ",ans);
    	}
    	return 0;
    }
    

    Sol2:

    #include<bits/stdc++.h>
    using namespace std;
    typedef long long ll;
    ll o,p[100011],sum[100011],len,ans,j;
    char a[100011];
    int main()
    {
        scanf("%lld",&o);
        while(o--)
        {
            memset(p,0,sizeof(p));
            memset(sum,0,sizeof(sum));
            scanf("%s",a+1);
            len=strlen(a+1);
            ans=0;j=0;
            for(ll i=1;i<len;i++)
            {
                while(j&&a[j+1]!=a[i+1])
                    j=p[j];
                if(a[j+1]==a[i+1])
                    j++;
                p[i+1]=j;
            }
            for(ll i=1;i<=len;i++)
                sum[i]=sum[p[i]]+1,
                ans+=sum[i];
            printf("%lld
    ",ans);
        }
        return 0;
    }

     推荐使用下面这个进行讲解

    #include<bits/stdc++.h>
    using namespace std;
    typedef long long ll;
    ll o,p[100011],sum[100011],len,ans,j;
    char a[100011];
    int main()
    {
        scanf("%lld",&o);
        while(o--)
        {
            memset(p,0,sizeof(p));
            memset(sum,0,sizeof(sum));
            scanf("%s",a+1);
            len=strlen(a+1);
            ans=0;j=0;
            for(ll i=1;i<len;i++)
            {
                while(j&&a[j+1]!=a[i+1])
                    j=p[j];
                if(a[j+1]==a[i+1])
                    j++;
                p[i+1]=j;
            }
            for (int i=1;i<=len;i++)
            sum[i]=1;
            for(ll i=len;i>=1;i--)
                sum[p[i]]+=sum[i];
            for (int i=1;i<=len;i++)
                ans=ans+sum[i];
            printf("%lld
    ",ans);
        }
        return 0;
    }
  • 相关阅读:
    ipv6 for openwrt odhcpd
    openwrt package Makefile
    openwrt 中个网络接口协议说明[转]
    openwrt Package aircrack-ng is missing dependencies for the following libraries:
    linux kernel 从cmdline 提取值
    js 上传文件进度条 [uboot使用]
    printk打印级别 [转]
    linux c 宏定义
    uboot 开发记录
    mac ssh scp命令
  • 原文地址:https://www.cnblogs.com/cutemush/p/12505574.html
Copyright © 2011-2022 走看看