Day51 | 动态规划 :区间DP 最长回文子序列&&多边形三角部分的最低得分

Day51 | 动态规划 :区间DP 最长回文子序列&&多边形三角部分的最低得分

动态规划应该如何学习?-CSDN博客

本次题解参考自灵神的做法,大家也多多支持灵神的题解

区间 DP:最长回文子序列【基础算法精讲 22】_哔哩哔哩_bilibili

动态规划学习:

1.思考回溯法(深度优先遍历)怎么写

注意要画树形结构图

2.转成记忆化搜索

看哪些地方是重复计算的,怎么用记忆化搜索给顶替掉这些重复计算

3.把记忆化搜索翻译成动态规划

基本就是1:1转换

文章目录

516.最长回文子序列

516. 最长回文子序列 - 力扣(LeetCode)

思路分析(子问题):

1.s和反转s求它俩的最长公共子序列

2.直接求

还是选或不选的思路

dfs(i,j)表示从i到j这段里面的最长回文子序列的长度

那就是看选或不选s[i]和s[j]

三种情况

选s[i]和s[j]

选s[i]不选s[j]

不选s[i]选s[j]

如果s[i]==s[j]

dfs(i,j)=dfs(i+1,j-1)+2;

如果s[i]!=s[j],那就是两个里面选个最大值

cpp 复制代码
dfs(i,j)=max(dfs(i+1,j),dfs(i,j-1));

递归边界

dfs(i,j) i>j  

说明i和j中间没有字符,是空串,返回0

dfs(i,j) i==j

说明i到j只有一个字符,这个字符组成一个子序列,长度为1,并且也是回文的,返回1

1.回溯 DFS

1.返回值和参数

dfs(i,j)表示从i到j这段里面的最长回文子序列的长度

cpp 复制代码
int dfs(int i,int j,string &s)

2.终止条件

对应递归边界

cpp 复制代码
        if(i>j)
            return 0;
        if(i==j)
            return 1;

3.本层逻辑

cpp 复制代码
        if(s[i]==s[j])
            return dfs(i+1,j-1,s)+2;
        else
            return max(dfs(i+1,j,s),dfs(i,j-1,s));

完整代码:

当然,这是超时的

cpp 复制代码
class Solution {
public:
    int dfs(int i,int j,string &s)
    {
        if(i>j)
            return 0;
        if(i==j)
            return 1;
        if(s[i]==s[j])
            return dfs(i+1,j-1,s)+2;
        else
            return max(dfs(i+1,j,s),dfs(i,j-1,s));
    }
    int longestPalindromeSubseq(string s) {
        
        return dfs(0,s.size()-1,s);
    }
};
cpp 复制代码
//lambda
class Solution {
public:
    int longestPalindromeSubseq(string s) {
        function<int(int,int)> dfs=[&](int i,int j)->int{
            if(i>j)
            return 0;
            if(i==j)
                return 1;
            if(s[i]==s[j])
                return dfs(i+1,j-1)+2;
            else
                return max(dfs(i+1,j),dfs(i,j-1));
        };
        return dfs(0,s.size()-1);
    }
};

2.记忆化搜索

就是搞一个哈希表dp,全都初始化为-1,每次返回前给哈希表dp赋值,碰到不是-1的那就是算过的,那就直接返回计算过的结果,不需要再次递归了

cpp 复制代码
class Solution {
public:
    int dfs(int i,int j,string &s,vector<vector<int>> &dp)
    {
        if(i>j)
            return 0;
        if(i==j)
            return 1;
        if(dp[i][j]!=-1)
            return dp[i][j];
        if(s[i]==s[j])
            return dp[i][j]=dfs(i+1,j-1,s,dp)+2;
        else
            return dp[i][j]=max(dfs(i+1,j,s,dp),dfs(i,j-1,s,dp));
    }
    int longestPalindromeSubseq(string s) {
        vector<vector<int>> dp(s.size(),vector<int>(s.size(),-1));
        return dfs(0,s.size()-1,s,dp);
    }
};
cpp 复制代码
//lambda
class Solution {
public:
    int longestPalindromeSubseq(string s) {
        vector<vector<int>> dp(s.size(),vector<int>(s.size(),-1));
        function<int(int,int)> dfs=[&](int i,int j)->int{
            if(i>j)
            return 0;
            if(i==j)
                return 1;
            if(dp[i][j]!=-1)
                return dp[i][j];
            if(s[i]==s[j])
                return dp[i][j]=dfs(i+1,j-1)+2;
            else
                return dp[i][j]=max(dfs(i+1,j),dfs(i,j-1));
        };
        return dfs(0,s.size()-1);
    }
};

3.1:1翻译为动态规划

1.确定dp数组以及下标的含义

dp[i][j]就是dfs(I,j)

2.确定递推公式

和dfs中也是对应的

cpp 复制代码
				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]);

3.dp数组如何初始化

都初始化为0即可

cpp 复制代码
vector<vector<int>> dp(s.size(),vector<int>(s.size(),0));
for(int i=0;i<s.size();i++)
	dp[i][i]=1;

4.确定遍历顺序

i由i+1推导来的,所以i需要倒序遍历

j由j-1推导来的,所以j需要正序遍历

为什么j从i+1开始?

因为j<i的时候都是空串,我们在递归中直接返回的0,这里咱们初始化的时候已经做了这个工作,就不需要再管了

cpp 复制代码
		for(int i=s.size()-1;i>=0;i--)
            for(int j=i+1;j<s.size();j++)
                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]);

完整代码

cpp 复制代码
class Solution {
public:
    int longestPalindromeSubseq(string s) {
        vector<vector<int>> dp(s.size(),vector<int>(s.size(),0));
        for(int i=0;i<s.size();i++)
            dp[i][i]=1;
        for(int i=s.size()-1;i>=0;i--)
            for(int j=i+1;j<s.size();j++)
                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][s.size()-1];
    }
};

1039.多边形三角剖分的最低得分

1039. 多边形三角剖分的最低得分 - 力扣(LeetCode)

区间 DP:最长回文子序列【基础算法精讲 22】_哔哩哔哩_bilibili

笔者就贴一下代码,思路啥的大家看灵神讲吧

1.回溯dfs

cpp 复制代码
class Solution {
public:
    int dfs(int i,int j,vector<int>& values)
    {
        if(i+1==j)
            return 0;
        int res=INT_MAX;
        for(int k=i+1;k<j;k++)
            res=min(res,dfs(i,k,values)+dfs(k,j,values)+values[i]*values[j]*values[k]);
        return res;
    }
    int minScoreTriangulation(vector<int>& values) {
        return dfs(0,values.size()-1,values);
    }
};

2.记忆化搜索

cpp 复制代码
class Solution {
public:
    int dfs(int i,int j,vector<int>& values,vector<vector<int>>& dp)
    {
        if(i+1==j)
            return 0;
        int res=INT_MAX;
        if(dp[i][j]!=-1)
            return dp[i][j];
        for(int k=i+1;k<j;k++)
            res=min(res,dfs(i,k,values,dp)+dfs(k,j,values,dp)+values[i]*values[j]*values[k]);
        return dp[i][j]=res;
    }
    int minScoreTriangulation(vector<int>& values) {
        vector<vector<int>> dp(values.size(),vector<int>(values.size(),-1));
        return dfs(0,values.size()-1,values,dp);
    }
};

3.动态规划

cpp 复制代码
class Solution {
public:
    int minScoreTriangulation(vector<int>& v) {
        int n = v.size();
        vector<vector<int>> f(n, vector<int>(n));
        for (int i = n - 3; i >= 0; i--) {
            for (int j = i + 2; j < n; j++) {
                f[i][j] = INT_MAX;
                for (int k = i + 1; k < j; k++) {
                    f[i][j] = min(f[i][j], f[i][k] + f[k][j] + v[i] * v[j] * v[k]);
                }
            }
        }
        return f[0][n - 1];
    }
};
相关推荐
学编程的小程4 小时前
LeetCode216
算法·深度优先
leeyayai_xixihah4 小时前
2.21力扣-回溯组合
算法·leetcode·职场和发展
01_4 小时前
力扣hot100——相交,回文链表
算法·leetcode·链表·双指针
萌の鱼4 小时前
leetcode 2826. 将三个组排序
数据结构·c++·算法·leetcode
Buling_04 小时前
算法-哈希表篇08-四数之和
数据结构·算法·散列表
AllowM4 小时前
【LeetCode Hot100】除自身以外数组的乘积|左右乘积列表,Java实现!图解+代码,小白也能秒懂!
java·算法·leetcode
RAN_PAND5 小时前
STL介绍1:vector、pair、string、queue、map
开发语言·c++·算法
fai厅的秃头姐!7 小时前
C语言03
c语言·数据结构·算法
lisanndesu7 小时前
动态规划
算法·动态规划
myprogramc7 小时前
十大排序算法
数据结构·算法·排序算法