标签:des style blog http color io os ar java
题目链接:
http://acm.hdu.edu.cn/showproblem.php?pid=4628
2 aa abb
1 2
给一字符串,每次可以移除一个回文串,求最少要多少步,可以把该串移空。
解题思路:
状态压缩dp
dp[i]表示i状态表示的字符串是否是回文的。
对每个状态,枚举除掉是回文串的子状态进行更新。
ps:对于状态i,枚举i的子状态可以这样写 for(int j=i;j;j=(i&(j-1))
代码:
//#include<CSpreadSheet.h> #include<iostream> #include<cmath> #include<cstdio> #include<sstream> #include<cstdlib> #include<string> #include<string.h> #include<cstring> #include<algorithm> #include<vector> #include<map> #include<set> #include<stack> #include<list> #include<queue> #include<ctime> #include<bitset> #include<cmath> #define eps 1e-6 #define INF 0x3f3f3f3f #define PI acos(-1.0) #define ll __int64 #define LL long long #define lson l,m,(rt<<1) #define rson m+1,r,(rt<<1)|1 #define M 1000000007 //#pragma comment(linker, "/STACK:1024000000,1024000000") using namespace std; int dp[1<<16],ans[1<<16]; char sa[22]; int n; int main() { //freopen("in.txt","r",stdin); //freopen("out.txt","w",stdout); int t; scanf("%d",&t); while(t--) { scanf("%s",sa); n=strlen(sa); dp[0]=false; for(int i=1;i<(1<<n);i++) { int l=0,r=n-1; bool flag=true; while(l<=r) { while(l<n&&(i&(1<<l))==0) l++; while(r>=0&&(i&(1<<r))==0) r--; if(l>r) break; if(sa[l]!=sa[r]) { flag=false; break; } l++; r--; } dp[i]=flag; //printf("i:%d %d\n",i,dp[i]); ans[i]=n; } ans[0]=0; for(int i=1;i<(1<<n);i++) { for(int j=i;j;j=(i&(j-1))) { if(dp[j]) ans[i]=min(ans[i],ans[i-j]+1); } } printf("%d\n",ans[(1<<n)-1]); } return 0; }
标签:des style blog http color io os ar java
原文地址:http://blog.csdn.net/cc_again/article/details/39964057