HDU-4628-Pieces

131 篇文章 0 订阅

这个题大意是给你一个字符串,每次你可以删除其回文子串,问你删除整个子串最少需要多少步。

思路:

1、最开始算出所有的回文串,我是直接枚举的,以位运算表示满足要求的回文子串状态

2、状态压缩,因为字符串长度不超过16,所以可以用数字的位表示。

3、搜索答案,我用的是背包的思想

最开始用DFS记忆化写的,一直超时,其实超时的原因是因为我判断当前字符串是否满足当前所枚举的回文串的时候是用循环的。后来发现其实可以用异或运算然后判断是否等于其差值得到,从而节省了时间,顺利过了~

代码:

#include<cstdio>
#include<cstring>
#include<iostream>
using namespace std;
const int maxn=1<<16;
const int inf=1<<29;
int isp[maxn],dp[maxn],len,n,cur;
char str[20];
void Init()
{
    cur=0;
    n=(1<<len)-1;
    for(int i=1;i<=n;i++)
    {
	int j=0,k=len-1;
	bool is=true;
	while(1)
	{
	    for(;j<len;j++)
		if(i&(1<<j))
		    break;
	    for(;k>=0;k--)
		if(i&(1<<k))
		    break;
	    if(str[j]!=str[k])
	    {
		is=false;
		break;
	    }
	    j++,k--;
	    if(j>k)
		break;
	}
	if(is)
	    isp[cur++]=i;
    }
}
int main()
{
    int T;
    scanf("%d",&T);
    while(T--)
    {
	scanf("%s",str);
	len=strlen(str);
	Init();
	if(cur==len)
	{
	    printf("%d\n",len);
	    continue;
	}
	memset(dp,0x7f,sizeof(dp));
	dp[0]=0;
	for(int i=cur-1;i>=0;i--)
	{
	    for(int j=n;j-isp[i]>=0;j--)
		if(j^isp[i]==j-isp[i])
		    dp[j]=min(dp[j],dp[j^isp[i]]+1);
	    if(dp[n]==1||dp[n]==2)
		break;
	}
	printf("%d\n",dp[n]);
    }
    return 0;
}


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值