题意:给出26个字母每个字母的价值,问字符串中有多少个满足以下条件的子串:
1.子串的第一个和最后一个相同
2.子串除了头和尾的其他字符的价值加起来和尾0
这题普通方法应该是O(n^2),但是在1e5的条件下肯定会超时,所以学习了大力学长奥义的O(n)方法。具体方法也说不清楚,看代码吧,很短,也容易看懂。只能说,相当奥义的方法。。
代码如下:
1 #include <stdio.h> 2 #include <algorithm> 3 #include <string.h> 4 #include <map> 5 using namespace std; 6 typedef long long ll; 7 8 char s[100000+5]; 9 ll val[100000+5]; 10 map<ll,int> M[26]; 11 12 int main() 13 { 14 for(int i=0;i<26;i++) scanf("%I64d",val+i); 15 scanf("%s",s+1); 16 int len = strlen(s+1); 17 18 ll pre = 0,ans = 0; 19 for(int i=1;i<=len;i++) 20 { 21 int m = s[i] - 'a'; 22 ans += M[m][pre]; 23 pre += val[m]; 24 M[m][pre] ++; 25 } 26 printf("%I64d\n",ans); 27 }