自学算法: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];
    }
相关推荐
tobias.b3 分钟前
408真题解析-2009-10-数据结构-排序
数据结构·算法·排序算法·408考研·408真题·真题解析
Zachary_zlc7 分钟前
有向无环图检测算法和关键路径算法
算法
你撅嘴真丑9 分钟前
素数回文数的个数 与 求分数序列和
算法
Wuliwuliii16 分钟前
贡献延迟计算DP
数据结构·c++·算法·动态规划·dp
ysn1111120 分钟前
简单多边形三角剖分---耳切法(含源码)
算法
e疗AI产品之路21 分钟前
一文介绍Philips DXL心电图算法
算法·pan-tompkins·心电分析
小袁顶风作案31 分钟前
leetcode力扣——135.分发糖果
算法·leetcode·职场和发展
橘颂TA42 分钟前
【Linux】从 “抢资源” 到 “优雅控场”:Linux 互斥锁的原理与 C++ RAII 封装实战(Ⅰ)
linux·运维·服务器·c++·算法
YGGP1 小时前
【Golang】LeetCode 19. 删除链表的倒数第 N 个节点
算法·leetcode·链表