动态规划解题套路框架

学习计划:

最长回文子序列

〇、必读文章

1、数据结构和算法学习指南(学习算法和刷题的框架思维)

  • 了解数据结构的操作和遍历(迭代or递归)
  • 从树刷起,结合框架思维,有利于理解(回溯、动态规划、分治等)

2、动态规划详解(动态规划解题套路框架)

  • 过程:递归的暴力解法 -> 带备忘录的递归解法 -> 非递归的动态规划解法
  • 特征:重叠子问题-->使用备忘录&自底向上,最优子结构,状态转移方程
  • 例题:凑零钱(dp[i] = min(dp[i], 1 + dp[i - coin]))

3、回溯算法详解(修订版)=DFS-----做选择

  • 3个问题:参数记录路径、选择列表(做选择和撤销选择)、结束条件
result = []
def backtrack(路径, 选择列表):
if 满足结束条件:
result.add(路径)
return for 选择 in 选择列表:
做选择
backtrack(路径, 选择列表)
撤销选择
  • 例题:八皇后问题、全排列问题

4、BFS 算法框架套路详解------求最短距离

  • 与DFS对比:使用队列,路径短,空间复杂度高
  • 问题本质:在图中找起点到终点的最近距离,队列入队访问邻接,记录访问过的
  • 例题:二叉树的最小高度(齐头并进,BFS时间复杂度低)、打开密码锁的最少次数(可以使用双向BFS,无需掌握)
  • 步骤:
// 计算从起点 start 到终点 target 的最近距离
int BFS(Node start, Node target) {
Queue<Node> q; // 核心数据结构
Set<Node> visited; // 避免走回头路 q.offer(start); // 将起点加入队列
visited.add(start);
int step = 0; // 记录扩散的步数 while (q not empty) {
int sz = q.size();
/* 将当前队列中的所有节点向四周扩散 */
for (int i = 0; i < sz; i++) {
Node cur = q.poll();
/* 划重点:这里判断是否到达终点 */
if (cur is target)
return step;
/* 将 cur 的相邻节点加入队列 */
for (Node x : cur.adj())
if (x not in visited) {
q.offer(x);
visited.add(x);
}
}
/* 划重点:更新步数在这里 */
step++;
}
}

5、我作了首诗,保你闭着眼睛也能写对二分查找

  • 防止两数相加产生溢出:mid = left + (right - left) / 2;
  • while里面是小于等于
  • 寻找左侧边界的二分:相等时不直接返回
int left_bound(int[] nums, int target) {
int left = 0, right = nums.length - 1;
// 搜索区间为 [left, right]
while (left <= right) {
int mid = left + (right - left) / 2;
if (nums[mid] < target) {
// 搜索区间变为 [mid+1, right]
left = mid + 1;
} else if (nums[mid] > target) {
// 搜索区间变为 [left, mid-1]
right = mid - 1;
} else if (nums[mid] == target) {
// 收缩右侧边界
right = mid - 1;
}
}
// 检查出界情况
if (left >= nums.length || nums[left] != target)
return -1;
return left;
}
  • 寻找右侧边界:left = mid + 1;「搜索区间」全都统一成两端都闭

6、我写了套框架,把滑动窗口算法变成了默写题

/* 滑动窗口算法框架 */
void slidingWindow(string s, string t) {
    unordered_map<char, int> need, window;
    for (char c : t) need[c]++;

    int left = 0, right = 0;
    int valid = 0; 
    while (right < s.size()) {
        // c 是将移入窗口的字符
        char c = s[right];
        // 右移窗口
        right++;
        // 进行窗口内数据的一系列更新
        ...

        /*** debug 输出的位置 ***/
        printf("window: [%d, %d)\n", left, right);
        /********************/

        // 判断左侧窗口是否要收缩
        while (window needs shrink) {
            // d 是将移出窗口的字符
            char d = s[left];
            // 左移窗口
            left++;
            // 进行窗口内数据的一系列更新
            ...
        }
    }
}
  • 例题:最小覆盖子串、字符串排列、找所有字母的异位词、最长无重复子串
  • 引出双指针问题:
    • 快慢指针(链表,有环)
    • 左右指针(数组或字符串问题,如二分,有序数组两数之和等于目标数,反转数组)

7、团灭 LeetCode 股票买卖问题

  • 通用解法:穷举,循环内部分别使用min和max;多次买卖用贪心/递归;限定交易次数递归参数传k递减;冷冻期改变递归函数dp的参数;手续费改变比较max的值
  • (状态机)多维DP数组的状态转移方程:针对几种股票类型分别使用db table解决,其中限定次数的使用三维dp表

8、经典动态规划:打家劫舍系列问题

  • 例题:标准动规、环形数组、二叉树打劫(相连的房子不能被打劫)
int res = Math.max(
// 不抢,去下家
dp(nums, start + 1),
// 抢,去下下家
nums[start] + dp(nums, start + 2)
);
  • 自顶向下:递归调用dp函数,自底向上:定义dp数组
  • 二叉树:抢孩子的左和右孩子
Map<TreeNode, Integer> memo = new HashMap<>();
public int rob(TreeNode root) {
if (root == null) return 0;
// 利用备忘录消除重叠子问题
if (memo.containsKey(root))
return memo.get(root);
// 抢,然后去下下家
int do_it = root.val
+ (root.left == null ?
0 : rob(root.left.left) + rob(root.left.right))
+ (root.right == null ?
0 : rob(root.right.left) + rob(root.right.right));
// 不抢,然后去下家
int not_do = rob(root.left) + rob(root.right); int res = Math.max(do_it, not_do);
memo.put(root, res);
return res;
}

9、一个方法解决三道区间问题

  • 区间问题:即线段问题,包括合并线段,找线段的交集;技巧:排序&画图
  • 区间覆盖问题:例删除被覆盖的区间--排序后分3种情况,相交区间合并,不相交则更新起点终点
  • 区间合并问题:排序,根据条件进行合并
  • 区间交集问题:讨论两个交集的各种情况,并根据大小决定更新哪个list的下标

10、一个函数秒杀 2Sum 3Sum 4Sum 问题

  • 两数之和:排序+双指针;去重数对:相等时指针相加
  • 三数之和:第一个数不重复&调用两数之和的方法
  • 四数之和:第一个数不重复&调用三数之和的方法
  • ……:编写递归函数,参数传递n表示几数之和

11、手把手带你刷二叉树(第一期)

  • 翻转二叉树:前序
  • (完美二叉树)填充二叉树节点的右侧指针:传递两个参数,前序node1.next = node2;
  • 二叉树展开为链表:没看懂

12、经典动态规划:0-1 背包问题

  • 状态和选择、dp数组的定义
dp[i][w] = max( 把物品i装进背包, 不把物品i装进背包 ) 
  • 状态转移方程
dp[i][w] = max(dp[i-1][w],  dp[i-1][w - wt[i-1]] + val[i-1] )
  • 代码
int knapsack(int W, int N, vector<int>& wt, vector<int>& val) {
// vector 全填入 0,base case 已初始化
vector<vector<int>> dp(N + 1, vector<int>(W + 1, 0));
for (int i = 1; i <= N; i++) {
for (int w = 1; w <= W; w++) {
if (w - wt[i-1] < 0) {
// 当前背包容量装不下,只能选择不装入背包
dp[i][w] = dp[i - 1][w];
} else {
// 装入或者不装入背包,择优
dp[i][w] = max(dp[i - 1][w - wt[i-1]] + val[i-1],
dp[i - 1][w]);
}
}
} return dp[N][W];
}

13、我用四个命令概括了 Git 的所有套路

  • 三个分区:working directory工作目录,stage/index area暂存区,commit history提交历史。
  • git status查看前两个分区,git log查看提交日志的内容
  • git checkout .:将工作目录修改过的文件恢复为暂存区的文件,不会删除在工作目录创建的新文件
  • git commit --amend:修改合并,不会新创建一个commit到history中
  • git reset X:对X文件的修改,不会提交到history中/把对a.txt的修改从stage区撤销,但依然保存在work dir中,变为unstage的状态。

14、提高刷题幸福感的小技巧

  • 递归函数debug时,在递归函数前后分别打印关键变量的值。

labuladong算法笔记总结的更多相关文章

  1. 学习Java 以及对几大基本排序算法(对算法笔记书的研究)的一些学习总结(Java对算法的实现持续更新中)

    Java排序一,冒泡排序! 刚刚开始学习Java,但是比较有兴趣研究算法.最近看了一本算法笔记,刚开始只是打算随便看看,但是发现这本书非常不错,尤其是对排序算法,以及哈希函数的一些解释,让我非常的感兴 ...

  2. 算法笔记--数位dp

    算法笔记 这个博客写的不错:http://blog.csdn.net/wust_zzwh/article/details/52100392 数位dp的精髓是不同情况下sta变量的设置. 模板: ]; ...

  3. 算法笔记--lca倍增算法

    算法笔记 模板: vector<int>g[N]; vector<int>edge[N]; ][N]; int deep[N]; int h[N]; void dfs(int ...

  4. 算法笔记--STL中的各种遍历及查找(待增)

    算法笔记 map: map<string,int> m; map<string,int>::iterator it;//auto it it = m.begin(); whil ...

  5. 算法笔记--priority_queue

    算法笔记 priority_queue<int>que;//默认大顶堆 或者写作:priority_queue<int,vector<int>,less<int&g ...

  6. 算法笔记--sg函数详解及其模板

    算法笔记 参考资料:https://wenku.baidu.com/view/25540742a8956bec0975e3a8.html sg函数大神详解:http://blog.csdn.net/l ...

  7. 算法笔记——C/C++语言基础篇(已完结)

    开始系统学习算法,希望自己能够坚持下去,期间会把常用到的算法写进此博客,便于以后复习,同时希望能够给初学者提供一定的帮助,手敲难免存在错误,欢迎评论指正,共同学习.博客也可能会引用别人写的代码,如有引 ...

  8. 算法笔记_067:蓝桥杯练习 算法训练 安慰奶牛(Java)

    目录 1 问题描述 2 解决方案   1 问题描述 问题描述 Farmer John变得非常懒,他不想再继续维护供奶牛之间供通行的道路.道路被用来连接N个牧场,牧场被连续地编号为1到N.每一个牧场都是 ...

  9. 算法笔记(c++)--回文

    算法笔记(c++)--回文 #include<iostream> #include<algorithm> #include<vector> using namesp ...

  10. 算法笔记(c++)--完全背包问题

    算法笔记(c++)--完全背包和多重背包问题 完全背包 完全背包不同于01背包-完全背包里面的东西数量无限 假设现在有5种物品重量为5,4,3,2,1  价值为1,2,3,4,5  背包容量为10 # ...

随机推荐

  1. k8s 如何关联pvc到特定的pv

    可以使用对 pv 打 label 的方式,具体如下: 创建 pv,指定 label $ cat nfs-pv2.yaml apiVersion: v1 kind: PersistentVolume # ...

  2. 9. 第八篇 kube-controller-manager安装及验证

    文章转载自:https://mp.weixin.qq.com/s?__biz=MzI1MDgwNzQ1MQ==&mid=2247483826&idx=1&sn=88f0cef6 ...

  3. 5G 与数字化转型的关系是怎样的?

    5G提供的是通信网络服务,数字化转型需要网络服务,但并不是必须使用5G网络,也就是说5G在数字化转型中并不是必虚的,但可以作为备选项,不过在某些行业比如农业.林业.牧业.港口.建筑等布设有线网络.无线 ...

  4. SpringBoot的starter到底是什么?

    前言 我们都知道,Spring的功能非常强大,但也有些弊端.比如:我们需要手动去配置大量的参数,没有默认值,需要我们管理大量的jar包和它们的依赖. 为了提升Spring项目的开发效率,简化一些配置, ...

  5. GitHub 供应链安全已支持 Dart 开发者生态

    通过 Dart 和 GitHub 团队的共同努力,自 10 月 7 日起,GitHub 的 Advisory Database (安全咨询数据库).Dependency Graph (依赖项关系图) ...

  6. Java递归查找层级文件夹下特定内容的文件

    递归查找文件 引言 或许是文件太多,想找某个文件又忘记放哪了;又或者是项目改造,需要将外部调用接口进行改造,项目太多,又无法排查.那么怎么快速找到自己想要的内容就是一件值得思考的事情了. 根据特定内容 ...

  7. Hbase之shell基本操作

    一.系统命令 启动hbase Shell ./bin/hbase shell 获取帮助 help 查询服务器状态 status 查询hbase版本 version 查询表 list 获取表描述 des ...

  8. How to get the return value of the setTimeout inner function in js All In One

    How to get the return value of the setTimeout inner function in js All In One 在 js 中如何获取 setTimeout ...

  9. 小程序返回上一级页面背景音乐报错 setBackgroundAudioState:fail title is nil!;

    小程序初始化在onLoad的时候加载了一次背景音乐. 如果此时报错是title必传.如果没有 会报错一次 setBackgroundAudioState:fail title is nil!; 这个都 ...

  10. day53-马踏棋盘

    马踏棋盘 1.算法优化的意义 算法是程序的灵魂,为什么有些程序可以在海量数据计算时,依旧保持高速计算? 编程中算法很多,比如八大排序算法(冒泡.选择.插入.快排.归并.希尔.基数.堆排序).查找算法. ...