2.3 字符串-添加最少字符使字符串都是回文字符串

给定一个字符串str,可以在str的任意地方添加字符,请返回在添加字符最少的情况下,让str整体都是回文字符串的结果

分析:
1)问题关键:如何以个例推整体,再由整体推回个例?
回文字符串问题:显然,当任意 [i… j]字符串为回文字符串时,其中的任意k使得[i+k…j-k] 同样也是回文字符串,这样可以推出结论:先假设字符串str[i…j]已经是回文字符串,那么不论是在前面或者后面再加一个字符,此时我们最多只需要再在另一头再加上这个字符就使整体再次成为字符串。(注:是否存在我单单只加一个字符不改变回文结构的情况,例:最特殊的,原字符串由同一个字符组成,此时再加一个该字符那肯定还是回文。)
2)由此可知,[i…j]最少添加几个字符的问题,我们可以根据几种情况去甩给子串。先分析,
① 当str[i] == str[j]时,问题直接交给[i+1…j-1],、
② 当str[i] != str[j] ,此时如何分析,?

1.头尾不相等,那么肯定头尾有一端是最终回文字符串的最外层的,是头还是尾我不管,先得出这个结论(因为要添加最少字符,所以让原本在最外层的还是呆在最外层肯定是最优方案之一)。
2.再来分析头尾问题,头尾至少有一个会成为最终字符串的最外层,那么肯定是谁添加的少谁来当那个外面的。那么此时问题就交给了 [i+1 … j] 以及 [i … j-1]这两个子串。这俩哪个子串成为回文串的需求比较小,我就选哪个,然后最外层就相应添加另一头的,(原字符串i…j两头不相等,注定了要分开,必须为其中一个重新分配合适对象,而且这个对象至少有一个是外来的,才能牵手成功!然后另一个暂时没分配的我先扔到子串里头,看看有没有合适的,没有我在按上面的步骤来考虑.)
3.通过dp数组来还原最终回文字符串结果:
首先取值肯定取 dp[0][str.length],然后逐一分情况讨论,
1)当 str[i] == str[j]时,还原出的回文串应当为:

str[i] + [i+1…j-1]上应该还原出来的回文串 + str[j]也就是str[i]

2)当两者不相等时,此时我们就需要分析,dp[i][j]上的值是从哪得到的,即dp[i+1][j] 和dp[i][j-1]的大小,当前者小时,dp值就从前者+1得到,那么此时,我们的决策是为i位置上的字符分配对象,将j仍进子串中,那么回文串通式为:

str[i] + [i+1....j]这里自己决策回文串 + str[i]

代码实现:

	//首先实现获取dp数组部分
public int[][] getDp(String str){
        char[] chas = str.toCharArray();

        int l = chas.length;

        int[][] dp = new int[l][l];

        for(int j = 1; j < l-1 ; j ++){
            if(chas[j-1] != chas[j]){
                dp[j-1][j] = 1;
            }
			//这里开始填写表格中的j往前部分
            for (int i = j - 2; i > -1; i--){
                if(chas[i] == chas[j]){
                    dp[i][j] = dp[i+1][j+1];
                }else{
                    dp[i][j] = Math.min(dp[i+1][j], dp[i][j-1]) + 1;
                }
            }
        }

        return dp;

    }

//这里是总实现代码
 public String main1(String str){
        if(str.length() < 2 || str.equals(null)){
            return str;
        }//特殊情况咱先过了
        int[][] dp = getDp(str);

        int i = 0;
        int j = str.length();
        char[] res = new char[str.length() + dp[0][str.length()]];
        int resl = 0;
        int resr = res.length;

        while (i <= j){
            if(str.charAt(i) == str.charAt(j)){
                res[resl] = str.charAt(i);
                res[resr] = str.charAt(i);
                i++;
                j--;
            }else if (dp[i][j-1] < dp[i+1][j]){
                res[resl] = str.charAt(j);
                res[resr] = str.charAt(j);
                j--;//最外层只用了原字符串的一个[j],另一个j是我们自己添加的
            }else{
                res[resl] = str.charAt(i);
                res[resr] = str.charAt(i);
                i++;///最外层只用了原字符串的一个[i],另一个i是我们自己添加的
            }
            resl++;
            resr++;
        }
        return String.valueOf(res);
    }
}

总结:
本题涉及到动态规划思想,每段字符串是否为回文的判断都交给了切割两端字符后的子串去解决,语言上比较好描述,但是我没能想出具体代码实现方法,最后研究代码实现过程仍然是十分简洁,getdp中的内for循环还是很容易想到,只是我一直纠结于任取一个0,j最终会交给 (j+0) >> 1 处的dp值来处理,刚开始一直纠结于这个地方是否可能还没有赋值,实际上这不存在。只要动笔随便哗啦几下就出来了,不能一直只用眼睛看。

  • 1
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值