当前位置: 首页 > article >正文

代码随想录二刷——动规篇章

dp 动态规划篇


dp篇二刷复习

  • 路径种数问题
    • 63.不同路径 II
  • 拆分问题
    • 343. 整数拆分
    • 96.不同的二叉搜索树
  • 背包问题
    • 0-1背包,最多选一个,选or不选
      • 406.分割等和子集
      • 1049. 最后一块石头的重量 II
      • 494.目标和
      • 474.一和零
    • 完全背包,每个物品可以选无限多次
      • 322.零钱兑换(用最少个数物品填满背包)
      • 279.完全平方数 (用最少的物品填满背包)
      • 518. 零钱兑换 II (组合的种数)
      • 377. 组合总和 Ⅳ (排列的种数)
      • 139.单词拆分(能不能由字典里的单词组成这个字符串)
    • 多重背包
  • 打家劫舍
    • 打家劫舍1
    • 打家劫舍 II
    • 打家劫舍3
      • 打家劫舍4
  • 股票问题
    • 121.买卖股票的最佳时机
    • 122. 买卖股票的最佳时机 II
    • 123. 买卖股票的最佳时机 III
    • 188. 买卖股票的最佳时机 IV
    • 309.最佳买卖股票时机含冷冻期
  • 线性dp、区间dp
    • 线性dp
      • 300. 最长递增子序列
      • 674. 最长递增子串
      • 1143.最长公共子序列
      • 718.最长公共子串
      • 53.最大子数组和
    • 编辑距离
      • 392.判断子序列
      • 115.不同的子序列
      • 583.两个字符串的删除操作
      • 72.编辑距离
    • 区间dp
      • 647.回文子串数目
      • 516.最长回文子序列


路径种数问题

63.不同路径 II

不同路径2
dp[i][j]表示到这个位置的走法,遍历到对应的obstacle位置如果是1,这个点就保持不变是0就行

拆分问题

343. 整数拆分

整数拆分
写的时候有点把一个整数拉开,去乘的感觉了,但是还是没写出来
怎么拉开去算,还需要一个算子j从1-i, 然后dp[i]就被拆成 j * (i-j), 一个个去试
背过吧还是

def integerBreak(self, n: int) -> int:
        dp = [0] * (n+1)
        dp[1] = 1
        for i in range(2,n+1):
            for j in range(1,i+1):
                dp[i] = max(dp[i], j*(i-j), j*dp[i-j])
        return dp[n]

96.不同的二叉搜索树

不同的二叉搜索树
怎么说呢,二叉搜索树是 如果不空 ,左<根<右
dp[i]就是i个结点的种数
然后用一个j,j -> 1-i分别去盘j当头的情况,也很好盘么,dp[i] += dp[j-1] * dp[i-j]
背过吧

def numTrees(self, n: int) -> int:
        dp = [0] * (n+1)     #dp[i]表示由i个结点组成的二叉搜索树有多少种
        dp[0] = 1
        for i in range(1,n+1):
            for j in range(1,i+1):
                dp[i] += dp[j-1] * dp[i-j]  # j 是1-i轮流做头结点,然后左边子树情况*右边子树情况
        return dp[n]

背包问题

0-1背包,最多选一个,选or不选

状态压缩之后的代码,遍历顺序固定

#  n, m  物品个数, 容积  
dp = [0] * (m+1)
for i in range(0, n):
	for j in range(m, weight[i]-1, -1):
		dp[j] = max(dp[j], dp[j - weight[i]] + value[i])

406.分割等和子集

dp[j]的数值一定是小于等于j的。
如果dp[j] == j 说明,集合中的子集总和正好可以凑成总和j,理解这一点很重要。
分割等和子集

1049. 最后一块石头的重量 II

最后一块石头的重量 II
解的所在的话,最后我们计算的dp[target], 就是target = sum // 2的容积内,最接近数组之和一半的数,由于我们是向下取整,另一半sum-dp[target], 一定稍微大一点;所以最后的解,sum-dp[target] - dp[tartget]

494.目标和

目标和
要填正负号,必定有整数部分,假设和为x;我们就设法找到填满容积为x的背包有多少种方法就行

def findTargetSumWays(self, nums: List[int], target: int) -> int:
        # 正数部分和是x, 负数部分和是sum-x, x-(sum-x) = target
        # dp[j]表示填满j容积有多少种方法
        if sum(nums) < abs(target) or (sum(nums)+target) % 2 != 0: return 0
        n, m = len(nums), (sum(nums)+target) // 2
        dp = [0] * (m+1)
        dp[0] = 1
        for i in range(n):
            for j in range(m, nums[i]-1, -1):
                dp[j] += dp[j-nums[i]]
        return dp[m]

解释一下dp[j]的推导,dp[j] += dp[j-nums[i]]
就是
不考虑nums[i]的情况下,填满容量为j的背包,有dp[j]种方法。

那么考虑nums[i]的话(只要搞到nums[i]),凑成dp[j]就有dp[j - nums[i]] 种方法。

例如:dp[j],j 为5,

已经有一个1(nums[i]) 的话,有 dp[4]种方法 凑成 dp[5]。
已经有一个2(nums[i]) 的话,有 dp[3]种方法 凑成 dp[5]。
已经有一个3(nums[i]) 的话,有 dp[2]种方法 凑成 dp[5]
已经有一个4(nums[i]) 的话,有 dp[1]种方法 凑成 dp[5]
已经有一个5 (nums[i])的话,有 dp[0]种方法 凑成 dp[5]
那么凑成dp[5]有多少方法呢,也就是把 所有的 dp[j - nums[i]] 累加起来。

474.一和零

一和零
这个容积有两个维度
dp[i][j]表示有i个0和j个1,最多能选几个
dp[i][j] = max(dp[i][j], dp[i-zeronum][j-onenum]+1) #不选,和 选

def findMaxForm(self, strs: List[str], m: int, n: int) -> int:
        zeronum, onenum = 0, 0
        dp = [[0] * (n+1) for _ in range(m+1)]
        for str in strs:
            zeronum = str.count('0')
            onenum = str.count('1')
            for i in range(m, zeronum-1, -1):
                for j in range(n, onenum-1, -1):
                    dp[i][j] = max(dp[i][j], dp[i-zeronum][j-onenum]+1)
        return dp[m][n]

背过吧

完全背包,每个物品可以选无限多次

完全背包和01背包不同之处就在于, 完全背包的内层遍历容积也要正序遍历
刚刚在复习01背包的时候,为了不让前面的物品被重复放入,倒序遍历容积;在完全背包可以重复放入,就正序遍历容积就行了

322.零钱兑换(用最少个数物品填满背包)

求怎么装能有的最大价值或者一个方案能最多选多少个物品的话

dp[j] = max(dp[j], dp[j-nums[i]]+val[i]/1)

如果说,求一个方案最少能用几个物品填满背包的话,

dp[j] = min(dp[j], dp[j-nums[i]]+1)

这道题代码

def coinChange(self, coins: List[int], amount: int) -> int:
        dp = [10001] * (amount+1)
        dp[0] = 0
        for c in coins:
            for j in range(c, amount+1):
                dp[j] = min(dp[j], dp[j-c]+1)
        return dp[amount] if dp[amount] != 10001 else -1

279.完全平方数 (用最少的物品填满背包)

279.完全平方数

518. 零钱兑换 II (组合的种数)

零钱兑换 II

如果说有多少种选法的话是组合数,dp[j]表示填满容积为j有多少种组合

dp[j] += dp[j-nums[i]]

并且dp[0] = 1表示说什么都不选也是一种

377. 组合总和 Ⅳ (排列的种数)

377. 组合总和 Ⅳ
组合的种数,就是不同顺序算一种
排列的种数,就是不同顺序的算多种

组合的种数,先遍历物品再遍历容积
排列的种数,先遍历容积再遍历物品
因为是完全背包,两者都要正序, 二者递推公式都是一样的

def combinationSum4(self, nums: List[int], target: int) -> int:
        n,m = len(nums), target
        dp = [0] * (m+1)
        dp[0] = 1
        for j in range(m+1):
            for i in range(n):
                if j >= nums[i]: dp[j] += dp[j-nums[i]]
        return dp[m]

进阶版的爬楼梯就是这个题的解法

139.单词拆分(能不能由字典里的单词组成这个字符串)

单词拆分
也把它看成一个完全背包,j也有一个容积的意味

def wordBreak(self, s: str, wordDict: List[str]) -> bool:
        dp = [False] * (len(s)+1)
        dp[0] = True
        for j in range(1,len(s)+1):
            for i in range(j): 
                if dp[i] and s[i:j] in wordDict:
                    dp[j] = True
        return dp[len(s)]

多重背包

01背包,每种物品一个,选还是不选
完全背包,每种物品无限多个,选或不选
多重背包的话,就是每件物品多了一个件数,我们在遍历的时候还有遍历一下件数

def test_multi_pack1():
    '''版本:改变遍历个数'''
    weight = [1, 3, 4]
    value = [15, 20, 30]
    nums = [2, 3, 2]
    bag_weight = 10

    dp = [0] * (bag_weight + 1)
    for i in range(len(weight)):
        for j in range(bag_weight, weight[i] - 1, -1):
            # 以上是01背包,加上遍历个数
            for k in range(1, nums[i] + 1):
                if j - k * weight[i] >= 0:
                    dp[j] = max(dp[j], dp[j - k * weight[i]] + k * value[i])

    print(" ".join(map(str, dp)))

if __name__ == '__main__':
    test_multi_pack1()

也可以把ki数组给展开,就是把物品数组给扩大,相应的weight和value数组加进重复的,然后用01背包

打家劫舍

打家劫舍1

打家劫舍
打家劫舍就是隔一个,当前这个不取和取,来一个最大值

def rob(self, nums: List[int]) -> int:
        if len(nums) < 2: return nums[0]
        dp = [0] * len(nums)
        dp[0] = nums[0]
        dp[1] = max(nums[0], nums[1])
        for i in range(2,len(nums)):
            dp[i] = max(dp[i-1], dp[i-2]+nums[i])
        return dp[len(nums)-1]

打家劫舍 II

打家劫舍2,就是首尾是一个圈,分别把不带首位和不带尾位的数组来做dp,找一个最大的结果
打家劫舍 II

def rob(self, nums: List[int]) -> int:
        if len(nums) == 1: return nums[0]
        if len(nums) == 2: return max(nums[0], nums[1])

        def func(nums):
            n = len(nums)
            dp = [0]*n
            dp[0], dp[1] = nums[0], max(nums[1], nums[0])
            for i in range(2,n):
                dp[i] = max(dp[i-1], dp[i-2]+nums[i])
            return dp[n-1]
        return max(func(nums[1:]), func(nums[:-1]))

打家劫舍3

这个就纯建议背过了
打家劫舍3

# Definition for a binary tree node.
# class TreeNode:
#     def __init__(self, val=0, left=None, right=None):
#         self.val = val
#         self.left = left
#         self.right = right
class Solution:
    def rob(self, root: Optional[TreeNode]) -> int:
        if not root: return 0
        if not root.left and not root.right: return root.val

        def dfs(node): #对于每一个结点我们选和不选有两种情况,我们拿一个元组去存(不选的值,选的值)
            if not node: return (0,0)
            left = dfs(node.left)
            right = dfs(node.right)
            
            #不选,选子树最大的
            val1 = max(left[0],left[1])+max(right[0],right[1])
            #选,本身结点的值加上不选左孩子,不选右孩子
            val2 = node.val + left[0] + right[0]
            return (val1,val2)
        res = dfs(root)
        return max(res)

打家劫舍4

打家劫舍4

股票问题

121.买卖股票的最佳时机

买卖股票的最佳时机
这道题就是找到任意两个数组元素之间的最大的正差,可以维护一个最小左值,res初始化为0,找一个最大的正差
1.维护最小左边界

def maxProfit(self, prices: List[int]) -> int:
        low = 10000
        res = 0
        for i in range(len(prices)):
            low = min(low, prices[i])
            res = max(res, prices[i]-low)
        return res

2.dp
这个题就是只能买卖一次
状态定义dp[i][0]第i天持有,dp[i][1]第i天不持有

def maxProfit(self, prices: List[int]) -> int:
        dp = [[0] * 2 for _ in range(len(prices))]
        dp[0][0] = -prices[0]
        dp[0][1] = 0
        for i in range(1,len(prices)):
            dp[i][0] = max(dp[i-1][0], -prices[i])
            dp[i][1] = max(dp[i-1][1], dp[i-1][0]+prices[i])
        return dp[len(prices)-1][1]

122. 买卖股票的最佳时机 II

122. 买卖股票的最佳时机 II

这个题就是说能反复买卖多次,但是一次最多持有一股
1.就是只收割正利润

def maxProfit(self, prices: List[int]) -> int:
        cnt = 0
        for i in range(1, len(prices)):
            w = prices[i] - prices[i-1]
            if w > 0: cnt += w
        return cnt 

2.dp
这个和上一个只能买卖一次的区别就在于,持有状态的状态转移

def maxProfit(self, prices: List[int]) -> int:
        m = len(prices)
        dp = [[0] * 2 for _ in range(m)]
        dp[0][0], dp[0][1] = -prices[0], 0
        for i in range(1, m):
            dp[i][0] = max(dp[i-1][0], dp[i-1][1]-prices[i]) #持有,上个时态就持有,上个时态不持有现在买
            dp[i][1] = max(dp[i-1][1], dp[i-1][0]+prices[i])
        return dp[m-1][1]
  1. 买卖股票的最佳时机含手续费
    买卖股票的最佳时机含手续费
    和买卖股票2 是一样的,就是持有的时候不止要支付price[i]还要支付手续费
def maxProfit(self, prices: List[int], fee: int) -> int:
        dp = [[0] * 2 for _ in range(len(prices))]
        dp[0][0] = -prices[0] - fee
        for i in range(1, len(prices)):
            dp[i][0] = max(dp[i-1][0], dp[i-1][1]-prices[i]-fee) #持有
            dp[i][1] = max(dp[i-1][1], dp[i-1][0] + prices[i]) #不持有
        return dp[-1][1]

123. 买卖股票的最佳时机 III

123. 买卖股票的最佳时机 III
最多可以买卖两次,这意味着可以买卖一次,可以买卖两次,也可以不买卖
所以说这个题每天有五个状态,
0 —— 没有操作
1 —— 第一次买入
2 —— 第一次卖出
3 —— 第二次买入
4 —— 第二次卖出

状态转移,要么是上一个状态就这样转移过来的,要么是做了买卖对应上一个状态转移
dp[i][0] = dp[i - 1][0]
dp[i][1] = max(dp[i - 1][1], dp[i - 1][0] - prices[i])
dp[i][2] = max(dp[i - 1][2], dp[i - 1][1] + prices[i])
dp[i][3] = max(dp[i - 1][3], dp[i - 1][2] - prices[i])
dp[i][4] = max(dp[i - 1][4], dp[i - 1][3] + prices[i])

def maxProfit(self, prices: List[int]) -> int:
        m = len(prices)
        dp = [[0] * 5 for _ in range(m)]
        dp[0][1], dp[0][3] = -prices[0], -prices[0]  #初始化第一次买和第二次买

        for i in range(1,m):
            dp[i][0] = dp[i-1][0]     #不做操作
            dp[i][1] = max(dp[i-1][1], dp[i-1][0]-prices[i]) #第一次买
            dp[i][2] = max(dp[i-1][2], dp[i-1][1]+prices[i]) #第一次卖
            dp[i][3] = max(dp[i-1][3], dp[i-1][2]-prices[i]) #第二次买
            dp[i][4] = max(dp[i-1][4], dp[i-1][3]+prices[i]) #第二次卖
        return dp[m-1][4]

188. 买卖股票的最佳时机 IV

买卖股票的最佳时机 IV
最多可以买卖k次
上一题抽象成规律,奇数买,偶数卖

def maxProfit(self, k: int, prices: List[int]) -> int:
        m = len(prices)
        if m == 0: return 0
        dp = [[0]*(2*k+1) for _ in range(m)]
        for j in range(1, 2*k, 2):
            dp[0][j] = -prices[0]
        for i in range(1,m):
            for j in range(0, 2*k-1, 2):
                dp[i][j+1] = max(dp[i-1][j+1], dp[i-1][j]-prices[i])   #奇数买
                dp[i][j+2] = max(dp[i-1][j+2], dp[i-1][j+1]+prices[i]) #偶数卖
        return dp[-1][2*k]

309.最佳买卖股票时机含冷冻期

最佳买卖股票时机含冷冻期
可以多次买卖,但是卖了之后,下一天不能买卖,进入冷冻期
这个也是挺难的,有四个状态
持有:
状态1:之前就买了之后没有操作,和今天买

不持有:
状态2:两天前就卖了,昨天是冷冻期,今天保持不持有
状态3:今天卖出了股票

今天是冷冻期:
状态4:今天是冷冻期那么昨天一定是卖出

状态转移

dp[i][0] = max(dp[i - 1][0], max(dp[i - 1][3], dp[i - 1][1]) - prices[i]);  #持有
dp[i][1] = max(dp[i - 1][1], dp[i - 1][3]); #不持有中的2
dp[i][2] = dp[i - 1][0] + prices[i];      #不持有中的3,今天卖掉
dp[i][3] = dp[i - 1][2];   #今天是冷冻期

代码

def maxProfit(self, prices: List[int]) -> int:
        dp = [[0] * 4 for _ in range(len(prices))]
        dp[0][0] = -prices[0]
        for i in range(1, len(prices)):
            dp[i][0] = max(dp[i-1][0], max(dp[i-1][3], dp[i-1][1])-prices[i])#持有股票 分为前一天就持有股票, 今天买(又分为前一天是冷冻期和保持不持有股票状态)
            dp[i][1] = max(dp[i-1][1], dp[i-1][3])#不持有股票状态中的保持卖出的状态,分为前一天就卖出,和前一天是冷冻期
            dp[i][2] = dp[i-1][0] + prices[i] #今天就卖出,前一天肯定是持有
            dp[i][3] = dp[i-1][2]   #冷冻期,前一天肯定是卖出
        return max(dp[-1][3], max(dp[-1][2], dp[-1][1]))

线性dp、区间dp

线性dp

300. 最长递增子序列

最长递增子序列
找连续递增子序列

def lengthOfLIS(self, nums: List[int]) -> int:
        n = len(nums)
        dp = [1] * n
        res = 1
        for i in range(1,n):
            for j in range(i):
                if nums[i] > nums[j]: dp[i] = max(dp[i], dp[j]+1)
            res = max(res, dp[i])
        return res

674. 最长递增子串

最长连续递增序列找连续递增子串

def findLengthOfLCIS(self, nums: List[int]) -> int:
        n = len(nums)
        dp = [1] * n
        for i in range(1,n):
            if nums[i] > nums[i-1]: dp[i] = dp[i-1] + 1   
        return max(dp)

1143.最长公共子序列

最长公共子序列
不相交的线,最长公共子序列的可视化版

def longestCommonSubsequence(self, text1: str, text2: str) -> int:
        m, n = len(text1), len(text2)
        dp = [[0] * (n+1) for _ in range(m+1)]
        for i in range(1,m+1):
            for j in range(1, n+1):
                if text1[i-1] == text2[j-1]: dp[i][j] = dp[i-1][j-1] + 1
                else: dp[i][j] = max(dp[i-1][j], dp[i][j-1])
        return dp[-1][-1]

718.最长公共子串

最长公共子串

def findLength(self, nums1: List[int], nums2: List[int]) -> int:
        m,n = len(nums1), len(nums2)
        dp = [[0]*(n+1) for _ in range(m+1)]
        res = 0
        for i in range(1,m+1):
            for j in range(1,n+1):
                if nums1[i-1] == nums2[j-1]: dp[i][j] = dp[i-1][j-1]+1
                res = max(res, dp[i][j])
        return res

总结,子串的状态转移来源只可能有一种,子序列的话要多

53.最大子数组和

最大子数组和
贪心
维护一个cnt, 当cnt<0就给之前累加的和清零,然后再从下个开始累加

def maxSubArray(self, nums: List[int]) -> int:
        cnt = 0
        res = nums[0]
        for i in range(len(nums)):
            cnt += nums[i]
            if cnt > res: res = cnt
            if cnt < 0: cnt = 0
        return res

dp

def maxSubArray(self, nums: List[int]) -> int:
        dp = [0] * len(nums)
        dp[0] = nums[0]
        res = nums[0]
        for i in range(1,len(nums)):
            dp[i] = max(dp[i-1]+nums[i], nums[i])
            res = max(res, dp[i])
        return res

编辑距离

392.判断子序列

判断子序列
dp[i][j]定义为以下标i-1, j-1为结尾的字符串S 和 字符串T 的相同子序列的长度

if (s[i - 1] == t[j - 1]),
那么dp[i][j] = dp[i - 1][j - 1] + 1;,因为找到了一个相同的字符,相同子序列长度自然要在dp[i-1][j-1]的基础上加1

if (s[i - 1] != t[j - 1]),
此时相当于t要删除元素,t如果把当前元素t[j - 1]删除,那么dp[i][j] 的数值就是 看s[i - 1]与 t[j - 2]的比较结果了,即:dp[i][j] = dp[i][j - 1];

def isSubsequence(self, s: str, t: str) -> bool:
        m, n = len(s), len(t)
        dp = [[0] * (n+1) for _ in range(m+1)]
        for i in range(1,m+1):
            for j in range(1,n+1):
                if s[i-1] == t[j-1]: dp[i][j] = dp[i-1][j-1] + 1
                else:
                    dp[i][j] = dp[i][j-1]
        return dp[m][n] == len(s)

115.不同的子序列

不同的子序列
dp[i][j]:以i-1为结尾的s子序列中出现以j-1为结尾的t的个数为dp[i][j]。

2.状态转移

这一类问题,基本是要分析两种情况

s[i - 1] 与 t[j - 1]相等
s[i - 1] 与 t[j - 1] 不相等

1) 当s[i - 1] 与 t[j - 1]相等时,dp[i][j]可以有两部分组成:

一部分是用s[i - 1]来匹配,那么个数为dp[i - 1][j - 1]。
一部分是不用s[i - 1]来匹配,个数为dp[i - 1][j]。

dp[i][j] = dp[i - 1][j - 1] + dp[i - 1][j];

例如: s:bagg 和 t:bag ,s[3] 和 t[2]是相同的,但是字符串s也可以不用s[3]来匹配,即用s[0]s[1]s[2]组成的bag。

当然也可以用s[3]来匹配,即:s[0]s[1]s[3]组成的bag。

2) 当s[i - 1] 与 t[j - 1]不相等时
dp[i][j]只有一部分组成,不用s[i - 1]来匹配,即:dp[i - 1][j]

3.base case

从递推公式dp[i][j] = dp[i - 1][j - 1] + dp[i - 1][j]; 和 dp[i][j] = dp[i - 1][j]; 中可以看出dp[i][0] 和dp[0][j]是一定要初始化的

dp[0][j]一定都是0,s如论如何也变成不了t

dp[i][0] 表示:以i-1为结尾的s可以随便删除元素,出现空字符串的个数。

那么dp[i][0]一定都是1,因为也就是把以i-1为结尾的s,删除所有元素,出现空字符串的个数就是1。

dp[0][0]应该是1,空字符串s,可以删除0个元素,变成空字符串t。

4.遍历顺序和解的位置

正序就行,解在dp[m][n]

def numDistinct(self, s: str, t: str) -> int:
        m, n = len(s), len(t)
        dp = [[0] * (n+1) for _ in range(m+1)]
        for i in range(m+1):
            dp[i][0] = 1
        for j in range(1, n+1):
            dp[0][j] = 0
        
        for i in range(1, m+1):
            for j in range(1, n+1):
                if s[i-1] == t[j-1]: dp[i][j] = dp[i-1][j-1] + dp[i-1][j]
                else: dp[i][j] = dp[i-1][j]
        return dp[m][n]

583.两个字符串的删除操作

两个字符串的删除操作

这个题就是编辑距离,可以增 可以删 的情况 少了一种替换的情况

def minDistance(self, word1: str, word2: str) -> int:
        m, n = len(word1), len(word2)
        dp = [[0] * (n+1) for _ in range(m+1)]
        for i in range(m+1):
            dp[i][0] = i
        for j in range(n+1):
            dp[0][j] = j
        
        for i in range(1,m+1):
            for j in range(1, n+1):
                if word1[i-1] == word2[j-1]: dp[i][j] = dp[i-1][j-1] 
                else: dp[i][j] = min(dp[i-1][j], dp[i][j-1]) + 1
        return dp[m][n]

72.编辑距离

编辑距离
word1[i-1] == word2[j-1],相等直接转化
word1[i-1] != word2[j-1], 增删改,操作+1

def minDistance(self, word1: str, word2: str) -> int:
        m, n = len(word1), len(word2)
        dp = [[0] * (n+1) for _ in range(m+1)]
        for i in range(m+1):
            dp[i][0] = i
        for j in range(n+1):
            dp[0][j] = j
        
        for i in range(1,m+1):
            for j in range(1, n+1):
                if word1[i-1] == word2[j-1]: dp[i][j] = dp[i-1][j-1]
                else:
                    dp[i][j] = min(dp[i-1][j], dp[i][j-1], dp[i-1][j-1]) + 1
        return dp[m][n]

区间dp

647.回文子串数目

回文子串
1.状态定义

布尔类型的dp[i][j]:表示区间范围[i,j] (注意是左闭右闭)的子串是否是回文子串,如果是dp[i][j]为true,否则为false。

2.状态转移

整体上是两种,就是s[i]与s[j]相等,s[i]与s[j]不相等这两种。

当s[i]与s[j]不相等,那没啥好说的了,dp[i][j]一定是false。

当s[i]与s[j]相等时,这就复杂一些了,有如下三种情况

情况一:下标i 与 j相同,同一个字符例如a,当然是回文子串
情况二:下标i 与 j相差为1,例如aa,也是回文子串
情况三:下标:i 与 j相差大于1的时候,例如cabac,此时s[i]与s[j]已经相同了,我们看i到j区间是不是回文子串就看aba是不是回文就可以了,那么aba的区间就是 i+1 与 j-1区间,这个区间是不是回文就看dp[i + 1][j - 1]是否为true。

3.base case

这道题开的数组是高宽都是n的,没有特殊的base case
初始化全为False就行

4.遍历顺序以及解的位置

由dp[i+1][j-1], 遍历顺序i是反序的,j是正序的

解的位置就是计数器的值

class Solution:
    def countSubstrings(self, s: str) -> int:
        n = len(s)
        dp = [[False] * n for _ in range(n)]
        cnt = 0
        for i in range(n-1, -1, -1):
            for j in range(i, n):
                if s[i] != s[j]: dp[i][j] = False
                else:
                    if j - i <= 1:
                        dp[i][j] = True
                        cnt += 1
                    elif dp[i+1][j-1]:
                        dp[i][j] = True
                        cnt += 1
        return cnt

516.最长回文子序列

最长回文子序列

这道题和上面的不同在于这道题是子序列,不需要连续
第二个区别是求的最长的,我们可以把dp数组的状态定义为长度

1.状态定义

dp[i][j]:字符串s在[i, j]范围内最长的回文子序列的长度为dp[i][j]。

这道题代码里我开的还是n行n列的数组,正好映射到s的下标那种

2.状态转移

在判断回文子串的题目中,关键逻辑就是看s[i]与s[j]是否相同

如果s[i]与s[j]相同,那么dp[i][j] = dp[i + 1][j - 1] + 2;

但是这题来说,j可不是从i开始,子串那题,我们j是从i开始,是因为有 a 这种单个字母需要被判回文子串的情况,而这个需要统计的是长度;我们j从i+1开始遍历

如果s[i]与s[j]不相同,说明s[i]和s[j]的同时加入 并不能增加[i,j]区间回文子串的长度,那么分别加入s[i]、s[j]看看哪一个可以组成最长的回文子序列。

加入s[j]的回文子序列长度为dp[i + 1][j]。

加入s[i]的回文子序列长度为dp[i][j - 1]。

那么dp[i][j]一定是取最大的,即:dp[i][j] = max(dp[i + 1][j], dp[i][j - 1]);

3.base case

由于上面我说的i j不可能重合,所以这题我们需要初始化dp[i][i] = 1, 根据定义单个的就是1, 当然了如果我们在代码里特判一下也行,那样的话j就可以从i开始遍历

其余的我们就初始化为0就行

4.遍历顺序以及解的位置

遍历顺序我们i倒序, j从i+1开始,正序遍历
解的位置就是dp[0][n], 从头到尾序列中最长回文序列是多长

class Solution:
    def longestPalindromeSubseq(self, s: str) -> int:
        n = len(s)
        dp = [[0] * n for _ in range(n)]
        for i in range(n):
            dp[i][i] = 1
        
        for i in range(n-1, -1, -1):
            for j in range(i+1, n):
                if s[i] == s[j]:
                    dp[i][j] = dp[i+1][j-1] + 2
                else:
                    dp[i][j] = max(dp[i+1][j], dp[i][j-1])
        return dp[0][n-1]

http://www.kler.cn/a/4854.html

相关文章:

  • 解决ERROR: This version of pnpm requires at least Node.js xxx 的问题
  • TDC-GP30 Data Sheet
  • Java多线程
  • react-quill 富文本组件编写和应用
  • imageio 图片转mp4 保存mp4
  • spingboot解析CSV文件
  • 快速搭建python爬虫管理平台
  • Vue计算属性
  • 【华为OD机试 2023最新 】 微服务的集成测试(C++ 100%)
  • 使用ControlNet 控制 Stable Diffusion
  • ACK One GitOps 最佳实践
  • 【算法总结】拓扑排序
  • 初探推荐系统-02
  • 科技大势怎么看 2023怎么干?
  • 2023美赛春季赛Y题数据思路代码 Understanding Used Sailboat Prices数学建模加赛
  • npm的常用命令
  • SQL语句优化的七种方法
  • Linux0.11 根文件系统挂载(四)
  • leetcode 珠玑妙算
  • 腾讯云服务器创建快照备份数据的方法
  • NoSQL数据库简介
  • USTB校园网一键登录开机自动登录
  • 行为识别SlowFast笔记--环境配置和Demo展示
  • labview节点公式节点反馈节点表达节点属性节点
  • 大数据框架之Hive:第10章 分区表和分桶表