代码随想录算法训练营DAY40\DAY41|C++动态规划Part.3|343.整数拆分、96.不同的二叉搜索树

本文主要是介绍代码随想录算法训练营DAY40\DAY41|C++动态规划Part.3|343.整数拆分、96.不同的二叉搜索树,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

DAY40休息日,本篇为DAY41的内容

文章目录

  • 343.整数拆分
    • 思路
      • dp含义
      • 递推公式(难点)
      • 初始化
      • 遍历顺序
      • 打印
    • CPP代码
    • 数学方法
    • 归纳证明法
  • 96.不同的二叉搜索树
    • 思路
      • dp含义
      • 递推公式
      • 初始化
      • 遍历顺序
      • 打印
    • CPP代码
    • 题目总结

343.整数拆分

力扣题目链接

文章讲解:343.整数拆分

视频讲解:动态规划,本题关键在于理解递推公式!| LeetCode:343. 整数拆分

状态:哥们儿把从1-10的整数拆分全写出来了,思路嘎嘎有,要想乘积最大,必须把数字全部拆成2或者3。但是,如何跟动态规划联系起来呢?

看完题解出来了,哥们儿那个属于是数学方法,但是差很多完整的思考,后文会给予证明。

我认为本题更适合使用数学方法来解决,也就是数学归纳法

看到这个题目,会疑问应该拆成两个还是三个还是四个呢?

之前我们说过,动态规划用来解包含重叠子问题的某问题,那么这里直接试试动态规划。

思路

在之前你走过拆分2-10的流程吗,找到什么感觉了吗?当我们在拆10的时候,可能把10拆成4、6(或者是其他的什么),我们之前也拆过4和6,自然拆成2、2、3、3。发现了吗,我们拆10这个问题包含了重叠子问题(拆4和6),所以试试动态规划吧!

dp含义

自然一点的想法:

dp[i]:拆分数字i,可以得到的最大乘积为dp[i]

递推公式(难点)

递推公式的重难点是什么呢?先思考我们如何才能得到dp[i]

首先我们拆分i,肯定首先拆成两个数字,也就是j(i - j)j是遍历1j的所有情况;

如果拆成3个数及3个数以上,我们就是j * dp[i - j],该式子的含义就是把i拆成三个或三个以上数字(因为dp[i - j]包含了所有的拆分方法,且至少拆成两个)

NOTE

为什么公式j * dp[i-j]是成分成2个以上数字呢?

首先我们要搞明白dp[i - j]的含义,拆分数字i-j,可以得到的最大乘积,他就暗含了将i-j拆分成两个或两个以上数字。

为什么j就不拆分了呢?

由于我们是从1开始一直遍历到j,所以已经暗含了拆分j的情况,因为我们还有旁边的dp[i-j]来给j补蛋呢!

综上所述, d p [ i ] = m a x ( d p [ i ] , m a x ( ( i − j ) ∗ j , d p [ i − j ] ∗ j ) ) dp[i] = max(dp[i], max((i - j) * j, dp[i - j] * j)) dp[i]=max(dp[i],max((ij)j,dp[ij]j))

为什么这里多了个max(dp[i], ...)呢?因为我们之前说过,我们需要遍历1~j的数,所以为了保留每个i当前最大成绩,与新遍历的j上下文做比较,保持dp[i]的最大值更新或者不更新。

初始化

本题中,我们只初始化dp[2]=1,因为严格来说,dp[0] dp[1]不应该初始化,因为这在我们定义dp数组含义时就确定了这俩是没有意义的数值,题中给定的n也是大于等于2的。

遍历顺序

还记得上面我们说把1遍历到j不,这里我们需要两层遍历,对于dp数组那肯定是从左到右了;关于j应该是从1开始,因为从0开始仍然是没有意义的,难道我们还把一个数拆成0和其他数吗?

for (int i = 3; i <= n; i++) {for (int j = 1; j < i - 1; j++) {dp[i] = max(dp[i], max((i - j) * j, dp[i - j] * j));}
}
//再优化一下子
for (int i = 3; i <= n ; i++) {for (int j = 1; j <= i / 2; j++) {dp[i] = max(dp[i], max((i - j) * j, dp[i - j] * j));}
}

打印

CPP代码

class Solution {
public:int integerBreak(int n) {vector<int> dp(n + 1, 0);dp[2] = 1;for (int i = 3; i <= n ; i++) {for (int j = 1; j <= i / 2; j++) {dp[i] = max(dp[i], max((i - j) * j, dp[i - j] * j));}}return dp[n];}
};

数学方法

归纳证明法

我在写1-10的最优拆分方案过程中,确实感受到了以下规律,这里是leetcode的官解归纳证明法

  • 第一步:证明最优的拆分方案中不会出现大于 4的整数。

假设出现了大于 4 4 4 的整数 x x x,由于$ 2(x−2)>x 在 在 x>4$时成立,将 x拆分成 2和 x−2可以增大乘积。因此最优的拆分方案中不会出现大于 4 的整数。

  • 第二步:证明最优拆分方案中可以不出现整数4

很明显,出现4的话可以用 2 × 2 2 \times2 2×2代替

此时,可知最优的拆分方案只会出现1、2、3三个数字

  • 第三步:证明 n ≥ 5 n \geq5 n5时,最优的拆分方案不会出现整数1.

n ≥ 5 n \geq5 n5时,如果出现了整数1,那么拆分中剩余的数的和为 n − 1 ≥ 4 n-1 \geq4 n14,对应这至少两个整数1和一个大于等于4的数。我们将其中任意一个整数 x x x加上1,乘积都会增大。

此时,可知当 n ≥ 5 n \geq5 n5时,最优拆分方案只有2和3

  • 第三步:证明当 n ≥ 5 n \geq5 n5时,最优的拆分方案中2的个数不会超过3个

如果出现了超过 3 个 2,那么将它们转换成 2 个 3,可以增大乘积,即 3 × 3 > 2 × 2 × 2 3 \times 3 > 2 \times 2 \times 2 3×3>2×2×2

综上, n ≥ 5 n \geq5 n5的最优拆分方案就唯一了,这是因为当最优的拆分方案中2的个数分别为0,1,2个时,就对应着n除以3的余数分别为0,2,1的情况。

并且 n = 4 n = 4 n=4时的拆分方案也可以放入分类讨论的结果;当 2 ≤ n ≤ 3 2\leq n \leq3 2n3时,只有唯一的拆分方案 1 × ( n − 1 ) 1 \times (n - 1) 1×(n1)

int integerBreak(int n) {int (n <= 3) {return n - 1;}int quotient = n / 3; //商int remainder = n % 3; //余数if (remainder == 0) {	 //能被3整除,全部拆成3return (int)pow(3, quotient);}else if (remainder == 1){	//余1return (int)pow(3, quotient - 1) * 4}else {	//余2return (int)pow(3, quotient) * 2;}
}

96.不同的二叉搜索树

力扣题目链接

文章讲解:96.不同的二叉搜索树

视频讲解:动态规划找到子状态之间的关系很重要!| LeetCode:96.不同的二叉搜索树

状态:这个动态规划我知道!有点明显。dp数组肯定是1维的,含义就是组成的不同BST的个数,关于递推公式我列举了n等于1-4时各能组成多少个BST,在写4时发现了大致的规律,但是没能力抽象成数学公式

思路

直观上,我们肯定是要把n=1、2、3直接拉出来比较的。

n=3时,

  • 当1为头结点,其右子树有两个结点,结点布局与n=2时一致
  • 当2为头结点,其左右子树都只有一个结点,布局和n=1一致
  • 当3位头结点,其左子树有两个节点,和n=2时一致

到这里我们就完全挖掘住了重叠的子问题。

dp[3] = 元素1为头结点搜索树的数量 + 元素2为头结点搜索树的数量 + 元素3为头结点搜索树的数量

元素1为头结点搜索树的数量 = 右子树有2个元素的搜索树数量 * 左子树有0个元素的搜索树数量

元素2为头结点搜索树的数量 = 右子树有1个元素的搜索树数量 * 左子树有1个元素的搜索树数量

元素3为头结点搜索树的数量 = 右子树有0个元素的搜索树数量 * 左子树有2个元素的搜索树数量

有2个元素的搜索树数量就是dp[2]

有1个元素的搜索树数量就是dp[1]

有0个元素的搜索树数量就是dp[0]

综上 d p [ 3 ] = d p [ 2 ] ∗ d p [ 0 ] + d p [ 1 ] ∗ d p [ 1 ] + d p [ 0 ] ∗ d p [ 2 ] dp[3] = dp[2] * dp[0] + dp[1] * dp[1] + dp[0] * dp[2] dp[3]=dp[2]dp[0]+dp[1]dp[1]+dp[0]dp[2].。

同理, d p [ 4 ] = d p [ 0 ] ∗ d p [ 3 ] + d p [ 1 ] ∗ d p [ 2 ] + d p [ 2 ] ∗ d p [ 1 ] + d p [ 3 ] ∗ d p [ 0 ] dp[4] = dp[0]*dp[3] + dp[1]*dp[2] + dp[2]*dp[1]+ dp[3]*dp[0] dp[4]=dp[0]dp[3]+dp[1]dp[2]+dp[2]dp[1]+dp[3]dp[0],其中 d p [ 3 ] dp[3] dp[3]可以继续拆分,很显然我们的递推公式应该写成:

d p [ i ] = ∑ j = 1 i d p [ j − 1 ] d p [ i − j ] dp[i] = \sum_{j=1}^{i}dp[j-1]dp[i-j] dp[i]=j=1idp[j1]dp[ij]j-1j为头结点左子树节点数量,i-j 为以j为头结点右子树节点数量。很明显需要两个循环,一个大循环i还有一个小循环j

dp含义

dp[i]:表示的是i个不同元素节点组成的二叉搜索树的个数为dp[i]

递推公式

上文分析中,已经给出了基本的递推公式

dp[i] += dp[以j为头结点左子树节点数量] * dp[以j为头结点右子树节点数量]

j相当于是头结点的元素,从1遍历到i为止。

所以递推公式:dp[i] += dp[j - 1] * dp[i - j]; ,j-1j为头结点左子树节点数量i-j 为以j为头结点右子树节点数量

初始化

从递推公式也可以看出来,本题其实只要初始化dp[0]就可以了,他是推导的基础。

从定义上来讲,空结点也是一颗二叉树,也是一颗二叉搜索树。

综上:dp[0] = 1

遍历顺序

首先一定是遍历节点数,从递归公式:dp[i] += dp[j - 1] * dp[i - j]可以看出,节点数为i的状态是依靠i之前节点数的状态。

那么遍历i里面每一个数作为头结点的状态,用j来遍历。

for (int i = 1; i <= n; i++){for (int j = 1; j <= i; i++) {dp[i] += dp[j - 1] * dp[i - j];}
}

打印

CPP代码

class Solution {
public:int numTrees(int n) {vector<int> dp(n + 1);dp[0] = 1;for (int i = 1; i <= n; i++) {for (int j = 1; j <= i; j++) {dp[i] += dp[j - 1] * dp[i - j];}}return dp[n];}
};
  • 时间复杂度: O ( n 2 ) O(n^2) O(n2)
  • 空间复杂度: O ( n ) O(n) O(n)

题目总结

本题我们用的是一种近似数学归纳法的推理。

在LeetCode官解中,给出了严格的数学证明,我认为这样的思考过程也是非常需要了解的。

  • LeetCode官解(必须手推一下!也不难!)

  • 卡塔兰数:

    • 卡塔兰数往往解决以下几类问题:
      • 有效的括号组合的数量。
      • 不同的二叉搜索树的数量。
      • 凸多边形划分成三角形的方法数量。
      • 在一个正方形格子图中从一角到另一角的路径数量,这些路径仅向上或向右移动,并且不越过对角线。
    • 递推公式 C 0 = 1 , C n + 1 = 2 ( 2 n + 1 ) n + 2 C n C_0=1, C_{n+1}=\frac{2(2n+1)}{n+2}C_n C0=1,Cn+1=n+22(2n+1)Cn
class Solution {
public:int numTrees(int n) {long long C = 1;for (int i = 0; i < n; ++i) {C = C * 2 * (2 * i + 1) / (i + 2);}return (int)C;}
};

这篇关于代码随想录算法训练营DAY40\DAY41|C++动态规划Part.3|343.整数拆分、96.不同的二叉搜索树的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/944272

相关文章

Windows下C++使用SQLitede的操作过程

《Windows下C++使用SQLitede的操作过程》本文介绍了Windows下C++使用SQLite的安装配置、CppSQLite库封装优势、核心功能(如数据库连接、事务管理)、跨平台支持及性能优... 目录Windows下C++使用SQLite1、安装2、代码示例CppSQLite:C++轻松操作SQ

C++中RAII资源获取即初始化

《C++中RAII资源获取即初始化》RAII通过构造/析构自动管理资源生命周期,确保安全释放,本文就来介绍一下C++中的RAII技术及其应用,具有一定的参考价值,感兴趣的可以了解一下... 目录一、核心原理与机制二、标准库中的RAII实现三、自定义RAII类设计原则四、常见应用场景1. 内存管理2. 文件操

C++中零拷贝的多种实现方式

《C++中零拷贝的多种实现方式》本文主要介绍了C++中零拷贝的实现示例,旨在在减少数据在内存中的不必要复制,从而提高程序性能、降低内存使用并减少CPU消耗,零拷贝技术通过多种方式实现,下面就来了解一下... 目录一、C++中零拷贝技术的核心概念二、std::string_view 简介三、std::stri

C++高效内存池实现减少动态分配开销的解决方案

《C++高效内存池实现减少动态分配开销的解决方案》C++动态内存分配存在系统调用开销、碎片化和锁竞争等性能问题,内存池通过预分配、分块管理和缓存复用解决这些问题,下面就来了解一下... 目录一、C++内存分配的性能挑战二、内存池技术的核心原理三、主流内存池实现:TCMalloc与Jemalloc1. TCM

HTML5 搜索框Search Box详解

《HTML5搜索框SearchBox详解》HTML5的搜索框是一个强大的工具,能够有效提升用户体验,通过结合自动补全功能和适当的样式,可以创建出既美观又实用的搜索界面,这篇文章给大家介绍HTML5... html5 搜索框(Search Box)详解搜索框是一个用于输入查询内容的控件,通常用于网站或应用程

Python实例题之pygame开发打飞机游戏实例代码

《Python实例题之pygame开发打飞机游戏实例代码》对于python的学习者,能够写出一个飞机大战的程序代码,是不是感觉到非常的开心,:本文主要介绍Python实例题之pygame开发打飞机... 目录题目pygame-aircraft-game使用 Pygame 开发的打飞机游戏脚本代码解释初始化部

C++ 函数 strftime 和时间格式示例详解

《C++函数strftime和时间格式示例详解》strftime是C/C++标准库中用于格式化日期和时间的函数,定义在ctime头文件中,它将tm结构体中的时间信息转换为指定格式的字符串,是处理... 目录C++ 函数 strftipythonme 详解一、函数原型二、功能描述三、格式字符串说明四、返回值五

Java中Map.Entry()含义及方法使用代码

《Java中Map.Entry()含义及方法使用代码》:本文主要介绍Java中Map.Entry()含义及方法使用的相关资料,Map.Entry是Java中Map的静态内部接口,用于表示键值对,其... 目录前言 Map.Entry作用核心方法常见使用场景1. 遍历 Map 的所有键值对2. 直接修改 Ma

C++作用域和标识符查找规则详解

《C++作用域和标识符查找规则详解》在C++中,作用域(Scope)和标识符查找(IdentifierLookup)是理解代码行为的重要概念,本文将详细介绍这些规则,并通过实例来说明它们的工作原理,需... 目录作用域标识符查找规则1. 普通查找(Ordinary Lookup)2. 限定查找(Qualif

Java调用C#动态库的三种方法详解

《Java调用C#动态库的三种方法详解》在这个多语言编程的时代,Java和C#就像两位才华横溢的舞者,各自在不同的舞台上展现着独特的魅力,然而,当它们携手合作时,又会碰撞出怎样绚丽的火花呢?今天,我们... 目录方法1:C++/CLI搭建桥梁——Java ↔ C# 的“翻译官”步骤1:创建C#类库(.NET