自学算法:03 一维动态规划

有些递归在展开计算时,总是重复调用同一个子问题的解,这种重复调用的递归变成动态规划会很有收益,而如果每次展开都是不同的解,或者重复调用的现象很少,那么没有改动态规划的必要。

所以任何动态规划问题都一定对应着一个有重复调用行为的递归,所以任何动态规划问题都可以从递归入手,逐渐实现动态规划的方法。

983. 最低票价

这道题用动态规划做其实可以用到暴力递归的关键代码,其实就是加了记录到dp数组的部分,让递归函数不再重复对子问题进行求解。

java 复制代码
public static int MAXN = 366;
	public static int[] dp = new int[MAXN];
    public static int[] duration = {1, 7, 30};
	public static int mincostTickets(int[] days, int[] costs) {
		int n = days.length;
		Arrays.fill(dp, 0, n + 1, Integer.MAX_VALUE);
		dp[n] = 0;
		for(int i = n; i >= 0; i--) {
			for(int k = 0, j = i; k < 3; k++) {
				while(j < days.length && duration[k] + days[i] > days[j]) j++;
				dp[i] = Math.min(dp[i], costs[k] + dp[j]);
			}
			
		}
		return dp[0];
	}

91. 解码方法

首先使用递归的方法暴力破解(超时),设置递归函数判定字符区间[0, i]中有多少种方法,也可以理解为一串数字的合法划分有多少种方法。首先判定当前下标对应字符是否为'0',因为题目给定解码中没有以'0'开头的解码,所以如果是'0'代表这个区间内没有任何一种方式可以解码,也就是之前的划分是有错的,所以返回方法数为0;如果不为'0',那就要进一步判定当前字符是否能和下一个字符构成不大于26的数字,如果没有这种情况,那就返回区间[0,(i + 1)]的方法数,反之,就再加上区间[0,(i + 2)]的方法数,其实这样就相当于划分当前字符为合法数字,还是划分相邻两个字符为合法数字。

java 复制代码
public int f(char[] s, int i) {
        if(i == s.length) return 1;
        int res;
        if(s[i] == '0') {
        	res = 0;
        }else {
        	res = f(s, i + 1);
        	if(i + 1 <s.length && (s[i] - '0') * 10 +(s[i+1] - '0') <= 26) {
        		res  += f(s, i + 2);
        	}
        }
        return res;
    }
    public int numDecodings(String s) {
        return f(s.toCharArray(),0);
    }

因为超时了,所以必须要改成动态规划的做法。会发现改动的部分很少,但是要注意,因为每次对dp数组的求解都完全依赖于它后面的值,所以必须初始化下标n的dp值为1,也可以理解能划分到下标n的字符就证明这个划分是对的,所以如果不对dp[n]初始化,就没有加上这个对的划分。

java 复制代码
public int numDecodings(String s) {
		int n = s.length();
        char[] chs = s.toCharArray();
        int[] dp = new int[n + 1];
        dp[n] = 1;
        if(chs[n - 1] != '0') dp[n - 1] = 1;
        for(int i = n - 2; i >= 0; i--) {
        	if(chs[i] == '0') dp[i] = 0;
        	else if((chs[i] - '0') * 10 + (chs[i + 1] - '0') <= 26) {
        		dp[i] = dp[i + 1] + dp[i + 2];
        	} else dp[i] = dp[i + 1];
        }
        return dp[0];
    }
相关推荐
无限码力12 分钟前
路灯照明问题
数据结构·算法·华为od·职场和发展·华为ode卷
嘻嘻哈哈樱桃13 分钟前
前k个高频元素力扣--347
数据结构·算法·leetcode
dorabighead13 分钟前
小哆啦解题记:加油站的奇幻冒险
数据结构·算法
Ritsu栗子30 分钟前
代码随想录算法训练营day35
c++·算法
好一点,更好一点40 分钟前
systemC示例
开发语言·c++·算法
卷卷的小趴菜学编程1 小时前
c++之List容器的模拟实现
服务器·c语言·开发语言·数据结构·c++·算法·list
林开落L1 小时前
模拟算法习题篇
算法
玉蜉蝣1 小时前
PAT甲级-1014 Waiting in Line
c++·算法·队列·pat甲·银行排队问题
我真不会起名字啊2 小时前
“深入浅出”系列之算法篇:(2)openCV、openMV、openGL
算法
南宫生2 小时前
力扣动态规划-7【算法学习day.101】
java·数据结构·算法·leetcode·动态规划