思路
可以枚举,三重循环,这么算,n需小于100,显然不行
每一个字母贡献的子串数,为其下标分别与前一个相同字母下标和后一个相同字母下标相减的绝对值相乘,再累计每一个字母即为答案
比如图中这个字符串,记算4号位a的贡献子串数
前一个a下标为1,后1个为6 (4-1)*(6-4)=6
解释一下 4-1 对应着bca(234号) 6-4对应着ab(45号)
分别以bca为起始位置,ba为终止位置的子串都只包含4号a,且是全部只包含4号a的子串
即为bca bcab ca cab a ab
注意初始,可能前面或者后面没有相同字母
代码
#include<iostream>
#include<string>
using namespace std;
const int N=1e5+5;
int pre[N],nex[N],a[27];
string s;
int main()
{
cin>>s;
s="0"+s; //下标从1开始
int n=s.length()-1; //长度
for(int i=1;i<=n;i++) //寻找前一个相同字符下标
{
int c=s[i]-'a';
pre[i]=a[c];
a[c]=i;
}
for(int i=0;i<26;i++) //初始化,可能后边没有相同字符 寻找前一个时,未初始,因为a数组默然为0
a[i]=n+1;
for(int i=n;i;i--) //寻找后一个相同字符下标
{
int c=s[i]-'a';
nex[i]=a[c];
a[c]=i;
}
long long int ans=0;
for(int i=1;i<=n;i++)
ans+=(long long)(i-pre[i])*(nex[i]-i);//相乘计算累加
cout<<ans;
return 0;
}
转载于这里