给一个串 s = s1s2... sn,你可以选定其一个非空子串,然后将该子串翻转。具体来说,若选定的子串区间为 [l, r](1 ≤ l ≤ r ≤ n),则翻转后该串变为 s1s2... sl - 1srsr - 1... slsr + 1... sn。
请你回答仅通过一次上述操作后,s 是否能变成回文串。串 s 是回文串,当且仅当它从左至右读出与从右至左读出完全相同,即 s1s2... sn = snsn - 1... s1。
解析:
我们只需要找到不对称的那一部分在进行每一位反转,这时候我们可以用 O(1)的时间复杂度进行优化。hash函数。
公式:get(l,r) - get(l,i)*p[r -i] + get2(l,i)*p[r-i] 其中 p[r - i ]是 进行了 p倍
#include<bits/stdc++.h>
#define ull int
typedef long long ll;
using namespace std;
const int mod = 1e9+7;
const int N=5e5+7;
int n,m,k;
char str[N];
int h1[N],h2[N],p[N];
void init() {
p[0]=1;
for(int i=1;i<=500000;i++) p[i]=(ll)p[i-1]*131%mod;
}
void init(int l,int r) {
h1[l-1]=0;
for(int i=l;i<=r;i++) {
h1[i]=((ll)h1[i-1]*131+str[i])%mod;
}
h2[r+1]=0;
for(int i=r;i>=l;i--) {
h2[i]=((ll)h2[i+1]*131+str[i])%mod;
}
}
int get1(int l,int r) {
return ((h1[r]-(ll)h1[l-1]*p[r-l+1]%mod)+mod)%mod;
}
int get2(int l,int r) {
return ((h2[l]-(ll)h2[r+1]*p[r-l+1])%mod+mod)%mod;
}
bool check1(int l,int r,int i) {
int hs1=(((ll)get1(l,r)-(ll)get1(l,i)*p[r-i]+(ll)get2(l,i)*p[r-i])%mod+mod)%mod;
int hs2=(((ll)get2(l,r)-(ll)get2(l,i)+(ll)get1(l,i))%mod+mod)%mod;
return hs1==hs2;
}
bool check2(int l,int r,int i) {
int hs1=(((ll)get1(l,r)-(ll)get1(i,r)+(ll)get2(i,r))%mod+mod)%mod;
int hs2=(((ll)get2(l,r)-(ll)get2(i,r)*p[i-l]+(ll)get1(i,r)*p[i-l])%mod+mod)%mod;
return hs1==hs2;
}
void solve()
{
scanf("%s",str+1);
n=strlen(str+1);
int l=1,r=n;
while(l<=r&&str[l]==str[r]) l++,r--;
if(l>r) {
printf("Yes\n");
return ;
}
init(l,r);
bool flag=false;
for(int i=l;i<r;i++) {
if(check1(l,r,i)) flag=true;
if(flag) break;
}
for(int i=r;i>l;i--) {
if(check2(l,r,i)) flag=true;
if(flag) break;
}
if(flag) printf("Yes\n");
else printf("No\n");
}
int main()
{
init();
int t;
cin >> t;
while(t--)
{
solve();
}
return 0;
}
时间复杂度为:O(n);