[关闭]
@TryMyEdge 2017-02-18T12:31:23.000000Z 字数 5691 阅读 859

专题八(字符串)

题解


题目

A Glass Beads

题目大意:

    一个长度为N(N<=10000)字符串本身的首尾相接(最右边的字符右边连接最左边的字符),问以哪一个字符为起点,能让这个字符串字典序最小。如果有多个起点,输出最左边的。
    T组数据。

解题思路:

    裸的最小表示法。
    小细节:把原字符串后面再接一个自己,处理的时候就不用取模了,在数据比较大的时候会快一点。
    时间复杂度o(T*N),空间复杂度o(N)。

AC代码:

  1. #include<iostream>
  2. #include<cstdio>
  3. #include<cstring>
  4. #define pq priority_queue
  5. #define Pi acos(-1.0)
  6. using namespace std;
  7. #define MOD 1000000007
  8. char s[20005];
  9. int main()
  10. {
  11. int t;
  12. int x,y,z,l;
  13. cin>>t;
  14. while(t--)
  15. {
  16. scanf("%s",s);
  17. l=strlen(s);
  18. for(int i=l;i<2*l;i++)
  19. s[i]=s[i-l];
  20. x=0;
  21. y=1;
  22. while(y<l && x<l)
  23. {
  24. z=0;
  25. while(z<l && s[x+z]==s[y+z]) z++;
  26. if(z==l)
  27. break;
  28. if(s[x+z]>s[y+z])
  29. x=x+z+1;
  30. else
  31. y=y+z+1;
  32. if(x==y)
  33. y++;
  34. }
  35. printf("%d\n",x+1);
  36. }
  37. return 0;
  38. }

B Oulipo

题目大意:

    给出字符串W(1<=|W|<=10000)和T(1<=|T|<=1000000),问T中有多少个W(可重叠)。
    T组数据。

解题思路:

    裸的kmp。
    找到第一个匹配后不退出,跑完整个T串,每次匹配长度为|W|的时候答案加一即可。
    时间复杂度o(T*(|W|+|T|)),空间复杂度o(|W|+|T|)。

AC代码:

  1. #include<iostream>
  2. #include<cstdio>
  3. #include<cstring>
  4. #define pq priority_queue
  5. #define Pi acos(-1.0)
  6. using namespace std;
  7. #define MOD 1000000007
  8. char w[10005],t[1000005];
  9. int nex[10005];
  10. void pre_kmp(char nows[],int nownex[])
  11. {
  12. nownex[0]=0;
  13. nownex[1]=0;
  14. int l=strlen(nows);
  15. for(int i=1;i<l;i++)
  16. {
  17. int j=nownex[i];
  18. while(j && nows[i]!=nows[j])
  19. j=nownex[j];
  20. if(nows[i]==nows[j])
  21. nownex[i+1]=j+1;
  22. else
  23. nownex[i+1]=0;
  24. }
  25. }
  26. int kmp(char nows1[],char nows2[],int nownex[])
  27. {
  28. pre_kmp(nows1,nownex);
  29. int j=0,nowans=0;
  30. int l1=strlen(nows1),l2=strlen(nows2);
  31. for(int i=0;i<l2;i++)
  32. {
  33. while(j && nows1[j]!=nows2[i])
  34. j=nownex[j];
  35. if(nows1[j]==nows2[i])
  36. j++;
  37. if(j==l1)
  38. {
  39. nowans++;
  40. j=nownex[j];
  41. }
  42. }
  43. return nowans;
  44. }
  45. int main()
  46. {
  47. int T;
  48. cin>>T;
  49. while(T--)
  50. {
  51. scanf("%s%s",w,t);
  52. printf("%d\n",kmp(w,t,nex));
  53. }
  54. return 0;
  55. }

C Revolving Digits

题目大意:

    把正整数N(N<10^100000)看成一个首尾相接的字符串,根据起点的不同数字也会改变,例如104在不同起点时分别为410、041、104。问这些不同的数字中,比N本身大、小或者相等的数字各有多少个。起点不同但是数字相同只算一种。
    T(1<=T<=50)组数据。

AC代码:

    exkmp的应用。
    把原串作为模式串,把两个原串接在一起作为匹配串,然后当前起点开始的后缀如果和模式串的匹配长度等于模式串长,说明相等。如果小于,那么比较不同的这一位,就知道比N大还是比N小了。
    接下来解决重复的问题。以第一个点为起点的时候,匹配长度一定等于模式串长,后面如果再出现匹配长度等于模式串长,说明这个字符串有循环节,那么从这个起点开始后面的起点表示的数字都和前面是重复的,所以直接结束,就能实现去重。
    小细节:仔细看一下去重的方法会发现等于N的数字永远是1(滑稽)。
    时间复杂度o(T*lg(N)),空间复杂度o(lg(N))。

AC代码:

  1. #include<iostream>
  2. #include<cstdio>
  3. #include<cstring>
  4. #define pq priority_queue
  5. #define Pi acos(-1.0)
  6. using namespace std;
  7. #define MOD 1000000007
  8. char s1[100005],s2[200005];
  9. int nex[100005],exlen[200005];
  10. int ans1,ans2;
  11. void pre_exkmp(char nows[],int nownex[])
  12. {
  13. int l=strlen(nows),x=0,p;
  14. nownex[0]=l;
  15. while(x<l && nows[x+1]==nows[x]) x++;
  16. nownex[1]=x;
  17. p=1;
  18. for(int i=2;i<l;i++)
  19. {
  20. if(nownex[i-p]+i<nownex[p]+p)
  21. nownex[i]=nownex[i-p];
  22. else
  23. {
  24. x=nownex[p]+p-i;
  25. if(x<0) x=0;
  26. while(x+i<l && nows[x]==nows[i+x]) x++;
  27. nownex[i]=x;
  28. p=i;
  29. }
  30. }
  31. }
  32. void exkmp(char nows1[],char nows2[],int nownex[],int nowex[])
  33. {
  34. pre_exkmp(nows1,nex);
  35. int l1=strlen(nows1),l2=strlen(nows2),p,x;
  36. ans1=ans2=0;
  37. nowex[0]=l1;
  38. p=0;
  39. for(int i=1;;i++)
  40. {
  41. if(nownex[i-p]+i<nowex[p]+p)
  42. nowex[i]=nownex[i-p];
  43. else
  44. {
  45. x=nowex[p]+p-i;
  46. if(x<0) x=0;
  47. while(x+i<l2 && x<l1 && nows1[x]==nows2[i+x]) x++;
  48. nowex[i]=x;
  49. p=i;
  50. }
  51. if(nowex[i]==l1)
  52. break;
  53. else
  54. {
  55. if(nows2[i+nowex[i]]<nows1[nowex[i]])
  56. ans1++;
  57. else
  58. ans2++;
  59. }
  60. }
  61. }
  62. int main()
  63. {
  64. int T;
  65. cin>>T;
  66. for(int t=1;t<=T;t++)
  67. {
  68. scanf("%s",s1);
  69. int nowl=strlen(s1);
  70. for(int i=0;i<2*nowl;i++)
  71. s2[i]=s1[i%nowl];
  72. s2[nowl*2]=0;
  73. exkmp(s1,s2,nex,exlen);
  74. printf("Case %d: %d 1 %d\n",t,ans1,ans2);
  75. }
  76. return 0;
  77. }

D 统计难题

题目大意:

    给出一张单词表,然后针对每个询问的前缀求含有该前缀的单词有多少个,一个单词的前缀包含自己本身。单词长度不超过10。

解题思路:

    裸字典树。
    建一棵字典树,更新每个单词的时候把沿途的点的权值都加一。询问的时候找到对应的那个点返回他的权值就行了。
    小细节:注意下如果这个点不存在的情况,返回0。掌握一下内存池(伪)的概念和写法。
    时间复杂度o(总字符数),空间复杂度o(26*总单词数)。

AC代码:

  1. #include<iostream>
  2. #include<cstdio>
  3. #include<cstring>
  4. #include<malloc.h>
  5. #define pq priority_queue
  6. #define Pi acos(-1.0)
  7. using namespace std;
  8. #define MOD 1000000007
  9. struct Node
  10. {
  11. int num;
  12. int son[26];
  13. }tre[1000005];
  14. int nums=0;
  15. char s[15];
  16. void add(char nows[],int x)
  17. {
  18. tre[x].num++;
  19. if(nows[0])
  20. {
  21. if(!tre[x].son[nows[0]-'a'])
  22. tre[x].son[nows[0]-'a']=++nums;
  23. add(nows+1,tre[x].son[nows[0]-'a']);
  24. }
  25. }
  26. int ask(char nows[],int x)
  27. {
  28. if(nows[0])
  29. {
  30. if(!tre[x].son[nows[0]-'a'])
  31. return 0;
  32. else
  33. return ask(nows+1,tre[x].son[nows[0]-'a']);
  34. }
  35. else
  36. return tre[x].num;
  37. }
  38. int main()
  39. {
  40. while(cin.getline(s,13),strlen(s))
  41. add(s,0);
  42. while(~scanf("%s",s))
  43. printf("%d\n",ask(s,0));
  44. return 0;
  45. }

E Palindrome

题目大意:

    给一个字符串S(|S|<=1000000),求它的最长回文子串的长度。
    多组数据。

解题思路:

    裸的manacher。
    时间复杂度o(|S|),空间复杂度o(|S|)。

AC代码:

  1. #include<iostream>
  2. #include<cstdio>
  3. #include<cstring>
  4. #include<malloc.h>
  5. #define pq priority_queue
  6. #define Pi acos(-1.0)
  7. using namespace std;
  8. #define MOD 1000000007
  9. char s1[1000005],s2[2000005];
  10. int len[2000005];
  11. int mlc(char nows[],int nowlen[])
  12. {
  13. int ans=0,mx=0,po,l=strlen(nows);
  14. for(int i=0;nows[i];i++)
  15. {
  16. if(mx>i)
  17. len[i]=min(mx-i+1,len[2*po-i]);
  18. else
  19. len[i]=1;
  20. while(i-len[i]>=0 && i+len[i]<l && nows[i-len[i]]==nows[i+len[i]])
  21. len[i]++;
  22. if(i+len[i]-1>mx)
  23. {
  24. mx=i+len[i]-1;
  25. po=i;
  26. }
  27. ans=max(ans,len[i]);
  28. }
  29. return ans-1;
  30. }
  31. int main()
  32. {
  33. int l,t=0;
  34. while(scanf("%s",s1),s1[0]!='E')
  35. {
  36. t++;
  37. l=2*strlen(s1);
  38. for(int i=0;i<l;i++)
  39. {
  40. if(i%2)
  41. s2[i]=s1[i/2];
  42. else
  43. s2[i]='#';
  44. }
  45. s2[l]='#';
  46. s2[l+1]=0;
  47. printf("Case %d: %d\n",t,mlc(s2,len));
  48. }
  49. return 0;
  50. }

F Safe Or Unsafe

题目大意:

    用0和1对一个字符串S中的字母编码,满足任意一个编码都不是另一个编码的前缀(保证无二义性),然后将这个字符串编码,问这个字符串的最小编码长度会不会超过m(m是整数)。
    n组数据。

解题思路:

    题目都提示了哈夫曼编码,如果弄懂哈夫曼编码的概念这题应该是很简单的。
    扫描一遍字符串处理出每个字母出现的次数,然后把次数不为0的字母提取出来进行哈夫曼编码,最后把每个字母的哈夫曼编码长度乘上出现次数加起来和m进行比较。
    一个节点有两个属性,内含字符长度和当前编码长度。每次找到内含字符长度最小的两个点进行合并,合并后的点的内含字符长度等于两个点的内含字符长度相加,合并后的当前编码长度等于两个点当前编码长度加上两个点的内含字符长度。预处理每种字母的内含字符长度为出现次数,当前编码长度为0。
    小细节:特判一下只出现了一种字母的情况。代码很短我就不解释原理了,关于合并的操作大家自己模拟一下就明白了。
    时间复杂度o(|S|+26*log(26)),空间复杂度o(|S|)。

AC代码:

  1. #include<iostream>
  2. #include<cstdio>
  3. #include<cstring>
  4. #include<queue>
  5. #define pq priority_queue
  6. #define Pi acos(-1.0)
  7. using namespace std;
  8. #define MOD 1000000007
  9. struct Node
  10. {
  11. int val,num;
  12. friend bool operator <(Node x1,Node x2)
  13. {
  14. return x1.num>x2.num;
  15. }
  16. }temp,now;
  17. int nums[30];
  18. pq <Node> q;
  19. char s[1000006];
  20. int main()
  21. {
  22. int t,k;
  23. cin>>t;
  24. while(t--)
  25. {
  26. memset(nums,0,sizeof(nums));
  27. scanf("%d",&k);
  28. scanf("%s",s);
  29. for(int i=0;s[i];i++)
  30. nums[s[i]-'a']++;
  31. for(int i=0;i<26;i++)
  32. {
  33. if(nums[i])
  34. {
  35. temp.val=0;
  36. temp.num=nums[i];
  37. q.push(temp);
  38. }
  39. }
  40. while(1)
  41. {
  42. temp=q.top();
  43. q.pop();
  44. if(q.empty())
  45. break;
  46. now=q.top();
  47. q.pop();
  48. temp.val+=temp.num+now.num+now.val;
  49. temp.num+=now.num;
  50. q.push(temp);
  51. }
  52. if(temp.num<=k && temp.val<=k)
  53. printf("yes\n");
  54. else
  55. printf("no\n");
  56. }
  57. return 0;
  58. }
添加新批注
在作者公开此批注前,只有你和作者可见。
回复批注