当前位置:   article > 正文

K、Strings in the Pocket(马拉车求回文子串个数板子)_k. strings in the pocket

k. strings in the pocket

马拉车+暴力

思路:s和t相同时,就是找s的回文子串的数量,

如果两个串相同,可以视为找回文串个数。如果不同,先判断删除左边连续相同部分和右边连续相同部分后能否通过反转使两串相等,如果不行结果为0,如果可行不断往两边延伸。

教训!!!拿string写容易被卡时长

tle代码

  1. #include <bits/stdc++.h>
  2. #include<iostream>
  3. #include<algorithm>
  4. #include<string.h>
  5. using namespace std;
  6. typedef long long ll;
  7. string a,b;
  8. ll Manacher(string s)//专门用来找有多少个回文子串的
  9. { //切记区别于找最长回文串的马拉车
  10. /*改造字符串*/
  11. string res="$#";
  12. for(int i=0;i<s.size();++i){
  13. res+=s[i];
  14. res+="#";
  15. }
  16. /*数组*/
  17. ll ans=0;
  18. //新建p数组,大小和t串一致,初始化为0 ,p[i]表示以t[i]为中心的回文串半径
  19. vector<int> P(res.size() , 0);
  20. int mi=0,right=0; //mi为最大回文串对应的中心点,right为该回文串能达到的最右端的值
  21. for(int i=1;i<res.size();++i){
  22. P[i]=right>i ?min(P[2*mi-i],right-i):1; //关键句,文中对这句以详细讲解
  23. while(res[i+P[i]]==res[i-P[i]])
  24. ++P[i];
  25. if(right<i+P[i]){ //超过之前的最右端,则改变中心点和对应的最右端
  26. right=i+P[i];
  27. mi=i;
  28. }
  29. ans+=P[i]/2;
  30. }
  31. return ans;
  32. }
  33. int getAns(string s1,string s2) {
  34. int n = s1.size();
  35. int l = 0, r = n - 1;
  36. while (s1[l] == s2[l]) l++;
  37. while (s1[r] == s2[r]) r--;
  38. int a = l, b = r;
  39. while (a <= r) {
  40. if (s1[a] != s2[b]) return 0;
  41. a++, b--;
  42. }
  43. int ans = 0;
  44. do {
  45. ans++;
  46. l--, r++;
  47. } while (s1[l] == s1[r] && l >= 0 && r < n);
  48. return ans;
  49. }
  50. int main()
  51. {
  52. int t;
  53. cin>>t;
  54. while(t--){
  55. cin>>a>>b;
  56. if(a==b) cout<<Manacher(a)<<endl;
  57. else printf("%d\n", getAns(a, b));
  58. }
  59. return 0;
  60. }

加速过得马拉车板子

  1. #include "bits/stdc++.h"
  2. using namespace std;
  3. typedef long long LL;
  4. const int MAXN = 2e6 + 5;
  5. char s1[MAXN << 1], s2[MAXN];
  6. int p[MAXN << 1];
  7. LL manacher(char* s, int* p) {
  8. int n = strlen(s);
  9. for (int i = n; i >= 0; i--) {
  10. s[i + 1 << 1] = s[i];
  11. s[i << 1 | 1] = '#';
  12. }
  13. n = n + 1 << 1;
  14. s[0] = '$';
  15. int k = 0; LL ans = 0;
  16. for (int i = 2; i < n; i++) {
  17. if (i >= k + p[k]) p[i] = 1;
  18. else p[i] = min(p[2 * k - i], p[k] + k - i);
  19. while (s[i + p[i]] == s[i - p[i]]) p[i]++;
  20. if (p[i] + i > p[k] + k) k = i;
  21. ans += p[i] >> 1;
  22. }
  23. return ans;
  24. }

然后我分别贴两份值得借鉴的人的代码,真是让我深有感触啊!!!

  1. //第一份代码,由于马拉车函数 在初始化和拉车的过程中都用了位运算加速 所以使得拉车拉的极快,优势十分明显,为后期int getAns代码爆破的粗糙节省了一大笔时间,简直是漂亮!!!
  2. #include "bits/stdc++.h"
  3. using namespace std;
  4. typedef long long LL;
  5. const int MAXN = 2e6 + 5;
  6. char s1[MAXN << 1], s2[MAXN];
  7. int p[MAXN << 1];
  8. LL manacher(char* s, int* p) {
  9. int n = strlen(s);
  10. for (int i = n; i >= 0; i--) {
  11. s[i + 1 << 1] = s[i];
  12. s[i << 1 | 1] = '#';
  13. }
  14. n = n + 1 << 1;
  15. s[0] = '$';
  16. int k = 0; LL ans = 0;
  17. for (int i = 2; i < n; i++) {
  18. if (i >= k + p[k]) p[i] = 1;
  19. else p[i] = min(p[2 * k - i], p[k] + k - i);
  20. while (s[i + p[i]] == s[i - p[i]]) p[i]++;
  21. if (p[i] + i > p[k] + k) k = i;
  22. ans += p[i] >> 1;
  23. }
  24. return ans;
  25. }
  26. int getAns(char* s1, char* s2) {
  27. int n = strlen(s1);
  28. int l = 0, r = n - 1;
  29. while (s1[l] == s2[l]) l++;
  30. while (s1[r] == s2[r]) r--;
  31. int a = l, b = r;
  32. while (a <= r) {
  33. if (s1[a] != s2[b]) return 0;
  34. a++, b--;
  35. }
  36. int ans = 0;
  37. do {
  38. ans++;
  39. l--, r++;
  40. } while (s1[l] == s1[r] && l >= 0 && r < n);
  41. return ans;
  42. }
  43. int main() {
  44. int t;
  45. scanf("%d", &t);
  46. while (t--) {
  47. scanf("%s%s", s1, s2);
  48. if (strcmp(s1, s2)) printf("%d\n", getAns(s1, s2));
  49. else printf("%lld\n", manacher(s1, p));
  50. }
  51. return 0;
  52. }

第二份代码

  1. //第二份代码,虽然拉车函数写的一般,但好就好在下一步爆破时充分利用了L,R的标记节省了一大笔时间,可以说这哥们在爆破时没少动脑子,因为爆破的得力扬长避短的解决了拉车函数的不足!!!很精彩!!!
  2. #include <bits/stdc++.h>
  3. #include<iostream>
  4. #include<algorithm>
  5. #include<cstring>
  6. #include<string>
  7. using namespace std;
  8. typedef long long ll;
  9. const int N=2000000+10;
  10. int t,n,m,k,p,l,r,u,v;
  11. ll ans,cnt,flag,temp,sum;
  12. char a[N],b[N];
  13. int P[N<<1];
  14. ll Manacher(string s)
  15. {
  16. /*改造字符串*/
  17. string res="$#";
  18. for(int i=0;i<s.size();++i){
  19. res+=s[i];
  20. res+="#";
  21. }
  22. /*数组*/
  23. ll ans=0;
  24. int mi=0,right=0; //mi为最大回文串对应的中心点,right为该回文串能达到的最右端的值
  25. for(int i=1;i<res.size();++i){
  26. P[i]=right>i ?min(P[2*mi-i],right-i):1; //关键句,文中对这句以详细讲解
  27. while(res[i+P[i]]==res[i-P[i]])
  28. ++P[i];
  29. if(right<i+P[i]){ //超过之前的最右端,则改变中心点和对应的最右端
  30. right=i+P[i];
  31. mi=i;
  32. }
  33. ans+=P[i]/2;
  34. }
  35. return ans;
  36. }
  37. int main()
  38. { int t;
  39. cin>>t;
  40. while(t--){
  41. //scanf("%s%s",a,b);
  42. cin>>a>>b;
  43. int len=strlen(a);
  44. for(l=0;l<len&&a[l]==b[l];l++);
  45. for(r=len-1;r>=0&&a[r]==b[r];r--);
  46. if(l>r){
  47. cout<<Manacher(a)<<endl;
  48. }else{
  49. flag=1;
  50. for(int i=l;i<=r;i++){
  51. if(a[i]!=b[r-i+l]){
  52. flag=0;
  53. break;
  54. }
  55. }
  56. if(flag){
  57. ans=1;
  58. while(l-1>=0&&r+1<=len-1&&a[l-1]==a[r+1])ans++,l--,r++;
  59. cout<<ans<<endl;
  60. }else{
  61. cout<<0<<endl;
  62. }
  63. }
  64. }
  65. return 0;
  66. }

最后感谢Angel&DemonSTZG的代码,蟹蟹大神。

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/知新_RL/article/detail/799581
推荐阅读
相关标签
  

闽ICP备14008679号