<题目链接>
题目大意:
给定字符串$S(|S|<=5000)$,下标由1开始。然后q个问题$(q\leq10^6)$,对于每个问题,给定L,R,回答区间[L,R]里有多少个回文串。
解题分析:
先预处理出任意区间的字符串是否是回文串,然后就是用区间DP,对每个区间,根据它的子区间的状态进行转移,这里要用到容斥的思想统计一下该区间回文子串的个数。$$dp[i][j]=dp[i+1][j]+dp[i][j-1]-dp[i+1][j-1]+pali[i][j]$$
#include <bits/stdc++.h> using namespace std; const int N = 5e3+5; bool pali[N][N]; int dp[N][N]; char str[N]; int main(){ scanf("%s",str+1); int n=strlen(str+1); for(int i=1;i<=n;i++)pali[i][i]=1,dp[i][i]=1; for(int i=n;i>=1;i--){ for(int j=i+1;j<=n;j++){ if(str[i]==str[j] && j-i==1)pali[i][j]=1; //特判一下长度为2的区间 else if(str[i]==str[j] && pali[i+1][j-1])pali[i][j]=1; } }//先用简单的区间DP,预处理出所有区间的子串是否是回文串 for(int i=n;i>=1;i--){ for(int j=i+1;j<=n;j++){ dp[i][j]=dp[i+1][j]+dp[i][j-1]-dp[i+1][j-1]+pali[i][j]; //用容斥进行该区间内回文子串数量的统计 } } int q;scanf("%d",&q); while(q--){ int l,r;scanf("%d%d",&l,&r); printf("%d\n",dp[l][r]); } }