Powered by:NEFU AB-IN
Link
Nowcoder A. M形字符串
-
题意
给一个长度为n的字符串(1<=n<=200000),他只包含小写字母 找到这个字符串多少个前缀是M形字符串. M形字符串定义如下: 他由两个相同的回文串拼接而来,第一个回文串的结尾字符和第二个字符串的开始字符可以重叠,也就是以下都是M形字符串. abccbaabccba(由abccba+abccba组成) abcbaabcba(有abcba+abcba组成) abccbabccba(由abccba+abccba组成组成,但是中间的1是共用的) a(一个单独字符也算)
-
思路 判断回文串——字符串正向+反向哈希 判断这个字符串是不是两个回文串拼接成的,只需要判断
即可 -
代码 #include <bits/stdc++.h>
using namespace std;
#define ULL unsigned long long
#define SZ(X) ((int) X.size())
const int P = 131;
const int N = 2e5 + 10;
ULL h[N], p[N], hr[N];
int main(){
p[0] = 1;
string s;
cin >> s;
int n = SZ(s);
s = " " + s;
for(int i = 1; i <= n; ++ i){
hr[i] = hr[i - 1] * P + s[n - i + 1] - '0';
h[i] = h[i - 1] * P + s[i] - '0';
p[i] = p[i - 1] * P;
}
auto get = [&] (ULL *h, int l, int r){
return h[r] - h[l - 1] * p[r - l + 1];
};
auto check = [&] (int l, int r){
return get(h, l, r) == get(hr, n - r + 1, n - l + 1);
};
int ans = 0;
for(int i = 1; i <= n; ++ i){
if(check(1, i) && check(1, (i + 1) / 2)) ans ++;
}
cout << ans;
return 0;
}
|