自学算法: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];
    }
相关推荐
鸽鸽程序猿3 分钟前
【算法】【优选算法】宽搜(BFS)中队列的使用
算法·宽度优先·队列
Jackey_Song_Odd3 分钟前
C语言 单向链表反转问题
c语言·数据结构·算法·链表
Watermelo6177 分钟前
详解js柯里化原理及用法,探究柯里化在Redux Selector 的场景模拟、构建复杂的数据流管道、优化深度嵌套函数中的精妙应用
开发语言·前端·javascript·算法·数据挖掘·数据分析·ecmascript
乐之者v12 分钟前
leetCode43.字符串相乘
java·数据结构·算法
A懿轩A1 小时前
C/C++ 数据结构与算法【数组】 数组详细解析【日常学习,考研必备】带图+详细代码
c语言·数据结构·c++·学习·考研·算法·数组
古希腊掌管学习的神1 小时前
[搜广推]王树森推荐系统——矩阵补充&最近邻查找
python·算法·机器学习·矩阵
云边有个稻草人1 小时前
【优选算法】—复写零(双指针算法)
笔记·算法·双指针算法
半盏茶香1 小时前
在21世纪的我用C语言探寻世界本质 ——编译和链接(编译环境和运行环境)
c语言·开发语言·c++·算法
忘梓.2 小时前
解锁动态规划的奥秘:从零到精通的创新思维解析(3)
算法·动态规划
tinker在coding4 小时前
Coding Caprice - Linked-List 1
算法·leetcode