动态规划
<h2 id="什么是动态规划">什么是动态规划</h2><p>动态规划,英文:Dynamic Programming,简称DP,如果某一问题有很多重叠子问题,使用动态规划是最有效的。</p>
<p>所以动态规划中每一个状态一定是由上一个状态推导出来的,<strong>这一点就区分于贪心</strong>,贪心没有状态推导,而是从局部直接选最优的,</p>
<p>例如:有N件物品和一个最多能背重量为W 的背包。第i件物品的重量是weight,得到的价值是value 。<strong>每件物品只能用一次</strong>,求解将哪些物品装入背包里物品价值总和最大。</p>
<p>动态规划中dp是由dp]推导出来的,然后取max(dp, dp] + value)。</p>
<p>但如果是贪心呢,每次拿物品选一个最大的或者最小的就完事了,和上一个状态没有关系。</p>
<p>所以贪心解决不了动态规划的问题。</p>
<h2 id="动态规划的解题步骤">动态规划的解题步骤</h2>
<p>状态转移公式(递推公式)是很重要,但动规不仅仅只有递推公式。</p>
<p><strong>对于动态规划问题,可以拆解为如下五步曲:</strong></p>
<ol>
<li>确定dp数组(dp table)以及下标的含义</li>
<li>确定递推公式</li>
<li>dp数组如何初始化</li>
<li>确定遍历顺序</li>
<li>举例推导dp数组</li>
</ol>
<h2 id="01背包问题">01背包问题</h2>
<p>题目描述:有n件物品和一个最多能背重量为w 的背包。第i件物品的重量是weight,得到的价值是value 。<strong>每件物品只能用一次</strong>,求解将哪些物品装入背包里物品价值总和最大。</p>
<p>每一件物品其实只有两个状态,取或者不取,所以可以使用回溯法搜索出所有的情况,那么时间复杂度就是$o(2^n)$,这里的n表示物品数量。</p>
<p><strong>所以暴力的解法是指数级别的时间复杂度。进而才需要动态规划的解法来进行优化!</strong></p>
<p>举一个例子:背包最大重量为4。</p>
<p>物品为:</p>
<table>
<thead>
<tr>
<th></th>
<th>重量</th>
<th>价值</th>
</tr>
</thead>
<tbody>
<tr>
<td>物品0</td>
<td>1</td>
<td>15</td>
</tr>
<tr>
<td>物品1</td>
<td>3</td>
<td>20</td>
</tr>
<tr>
<td>物品2</td>
<td>4</td>
<td>30</td>
</tr>
</tbody>
</table>
<p>问背包能背的物品最大价值是多少?</p>
<h3 id="二维dp数组">二维dp数组</h3>
<ol>
<li>确定dp数组以及下标的含义</li>
</ol>
<p>对于背包问题,有一种写法, 是使用二维数组,即<strong>dp 表示从下标为的物品里任意取,放进容量为j的背包,价值总和最大是多少</strong>。</p>
<p><img src="https://seven97-blog.oss-cn-hangzhou.aliyuncs.com/imgs/202404271507823.png" alt="image-20240427150740757" loading="lazy"></p>
<ol start="2">
<li>确定递推公式</li>
</ol>
<p>再回顾一下dp]的含义:从下标为的物品里任意取,放进容量为j的背包,价值总和最大是多少。</p>
<p>那么可以有两个方向推出来dp,</p>
<ul>
<li><strong>不放物品i</strong>:由dp推出,即背包容量为j,里面不放物品i的最大价值,此时dp就是dp。(其实就是当物品i的重量大于背包j的重量时,物品i无法放进背包中,所以背包内的价值依然和前面相同。)</li>
<li><strong>放物品i</strong>:由dp]推出,dp] 为背包容量为j - weight的时候不放物品i的最大价值,那么dp] + value (物品i的价值),就是背包放物品i得到的最大价值</li>
</ul>
<p>所以递归公式: dp = max(dp, dp] + value);</p>
<ol start="3">
<li>dp数组如何初始化</li>
</ol>
<p><strong>关于初始化,一定要和dp数组的定义吻合,否则到递推公式的时候就会越来越乱</strong>。</p>
<p>首先从dp的定义出发,如果背包容量j为0的话,即dp,无论是选取哪些物品,背包价值总和一定为0。如图:</p>
<p><img src="https://seven97-blog.oss-cn-hangzhou.aliyuncs.com/imgs/202404271510433.png" alt="image-20240427151020339" loading="lazy"></p>
<p>在看其他情况。</p>
<p>状态转移方程 dp = max(dp, dp] + value); 可以看出i 是由 i-1 推导出来,那么i为0的时候就一定要初始化。</p>
<p>dp,即:i为0,存放编号0的物品的时候,各个容量的背包所能存放的最大价值。</p>
<p>那么很明显当 j < weight的时候,dp 应该是 0,因为背包容量比编号0的物品重量还小。</p>
<p>当j >= weight时,dp 应该是value,因为背包容量放足够放编号0物品。</p>
<p>代码初始化如下:</p>
<pre><code class="language-java">for (int j = 0 ; j < weight; j++) {// 当然这一步,如果把dp数组预先初始化为0了,这一步就可以省略,但很多同学应该没有想清楚这一点。
dp = 0;
}
// 正序遍历
for (int j = weight; j <= bagweight; j++) {
dp = value;
}
</code></pre>
<p>此时dp数组初始化情况如图所示:</p>
<p><img src="https://seven97-blog.oss-cn-hangzhou.aliyuncs.com/imgs/202404271511457.png" alt="image-20240427151108396" loading="lazy"></p>
<p>dp 和 dp 都已经初始化了,那么其他下标应该初始化多少呢?</p>
<p>其实从递归公式: dp = max(dp, dp] + value); 可以看出dp 是由左上方数值推导出来了,那么 其他下标初始为什么数值都可以,因为都会被覆盖。</p>
<p><strong>初始-1,初始-2,初始100,都可以!</strong></p>
<p>但只不过一开始就统一把dp数组统一初始为0,更方便一些。如图:</p>
<p><img src="https://seven97-blog.oss-cn-hangzhou.aliyuncs.com/imgs/202404271511333.png" alt="image-20240427151142267" loading="lazy"></p>
<ol start="4">
<li>确定遍历顺序</li>
</ol>
<p>在如下图中,可以看出,有两个遍历的维度:物品与背包重量</p>
<p><img src="https://seven97-blog.oss-cn-hangzhou.aliyuncs.com/imgs/202404271512812.png" alt="image-20240427151229720" loading="lazy"></p>
<p>那么问题来了,<strong>先遍历 物品还是先遍历背包重量呢?</strong></p>
<p><strong>其实都可以!! 但是先遍历物品更好理解</strong>。</p>
<p><strong>理解递归的本质和递推的方向</strong>。</p>
<p>dp = max(dp, dp] + value); 递归公式中可以看出dp是靠dp和dp]推导出来的。</p>
<p>先遍历物品,再遍历背包:</p>
<pre><code class="language-text">// weight数组的大小 就是物品个数
for(int i = 1; i < weight.size(); i++) { // 遍历物品
for(int j = 0; j <= bagweight; j++) { // 遍历背包容量
if (j < weight) dp = dp;
else dp = max(dp, dp] + value);
}
}
</code></pre>
<p>dp和dp] 都在dp的左上角方向(包括正上方向),那么先遍历物品,再遍历背包的过程如图所示:</p>
<p><img src="https://seven97-blog.oss-cn-hangzhou.aliyuncs.com/imgs/202404271514396.png" alt="" loading="lazy"></p>
<p>先遍历背包,再遍历物品:</p>
<pre><code class="language-java">// weight数组的大小 就是物品个数
for(int j = 0; j <= bagweight; j++) { // 遍历背包容量
for(int i = 1; i < weight.size(); i++) { // 遍历物品
if (j < weight) dp = dp;
else dp = max(dp, dp] + value);
}
}
</code></pre>
<p>先遍历背包,再遍历物品的过程如图:</p>
<p><img src="https://seven97-blog.oss-cn-hangzhou.aliyuncs.com/imgs/202404271515988.png" alt="" loading="lazy"></p>
<p><strong>可以看出,虽然两个for循环遍历的次序不同,但是dp所需要的数据就是左上角,根本不影响dp公式的推导!</strong></p>
<p>但先遍历物品再遍历背包这个顺序更好理解。</p>
<ol start="5">
<li>举例推导dp数组</li>
</ol>
<p>来看一下对应的dp数组的数值,如图</p>
<p><img src="https://seven97-blog.oss-cn-hangzhou.aliyuncs.com/imgs/202404271516971.png" alt="" loading="lazy"></p>
<p>最后的答案就是dp</p>
<h3 id="一维dp数组滚动数组">一维dp数组(滚动数组)</h3>
<p>对于背包问题其实状态都是可以压缩的。</p>
<p>在使用二维数组的时候,递推公式:dp = max(dp, dp] + value);</p>
<p><strong>其实可以发现如果把dp那一层拷贝到dp上,表达式完全可以是:dp = max(dp, dp] + value);</strong></p>
<p><strong>与其把dp这一层拷贝到dp上,不如只用一个一维数组了</strong>,只用dp(一维数组,也可以理解是一个滚动数组)。</p>
<p>这就是滚动数组的由来,需要满足的条件是上一层可以重复利用,直接拷贝到当前层。</p>
<p><strong>dp 表示从下标为的物品里任意取,放进容量为j的背包,价值总和最大是多少</strong>。</p>
<p>动规五部曲分析如下:</p>
<ol>
<li>确定dp数组的定义</li>
</ol>
<p>在一维dp数组中,dp表示:容量为j的背包,所背的物品价值可以最大为dp。</p>
<ol start="2">
<li>一维dp数组的递推公式</li>
</ol>
<p>dp为 容量为j的背包所背的最大价值,那么如何推导dp呢?</p>
<p>dp可以通过dp]推导出来,dp]表示容量为j - weight的背包所背的最大价值。</p>
<p>dp] + value 表示 容量为【j - 物品i重量】的背包 加上 物品i 的价值。(也就是容量为j的背包,放入物品i了之后的价值即:dp)</p>
<p>此时dp有两个选择,一个是取自己dp 相当于 二维dp数组中的dp,即不放物品i,一个是取dp] + value,即放物品i,指定是取最大的,毕竟是求最大价值,</p>
<p>所以递归公式为:</p>
<pre><code class="language-java">dp = max(dp, dp] + value);
</code></pre>
<p>可以看出相对于二维dp数组的写法,就是把dp中i的维度去掉了。</p>
<ol start="3">
<li>一维dp数组如何初始化</li>
</ol>
<p>dp表示:容量为j的背包,所背的物品价值可以最大为dp,那么dp就应该是0,因为背包容量为0所背的物品的最大价值就是0。</p>
<p>那么dp数组除了下标0的位置,初始为0,其他下标应该初始化多少呢?</p>
<p>看一下递归公式:dp = max(dp, dp] + value);</p>
<p>dp数组在推导的时候一定是取价值最大的数,如果题目给的价值都是正整数那么非0下标都初始化为0就可以了。</p>
<p><strong>这样才能让dp数组在递归公式的过程中取的最大的价值,而不是被初始值覆盖了</strong>。</p>
<p>那么我假设物品价值都是大于0的,所以dp数组初始化的时候,都初始为0就可以了。</p>
<ol start="4">
<li>一维dp数组遍历顺序</li>
</ol>
<p>代码如下:</p>
<pre><code class="language-java">for(int i = 0; i < weight.size(); i++) { // 遍历物品
for(int j = bagWeight; j >= weight; j--) { // 遍历背包容量
dp = max(dp, dp] + value);
}
}
</code></pre>
<p><strong>这里发现和二维dp的写法中,遍历背包的顺序是不一样的!</strong>二维dp遍历的时候,背包容量是从小到大,而一维dp遍历的时候,背包是从大到小。为什么呢?</p>
<p><strong>倒序遍历是为了保证物品i只被放入一次!</strong>。但如果一旦正序遍历了,那么物品0就会被重复加入多次!</p>
<p>举一个例子:物品0的重量weight = 1,价值value = 15</p>
<p>如果正序遍历</p>
<ul>
<li>
<p>dp = dp] + value = 15</p>
</li>
<li>
<p>dp = dp] + value = 30</p>
</li>
</ul>
<p>此时dp就已经是30了,意味着物品0,被放入了两次,所以不能正序遍历。</p>
<p>为什么倒序遍历,就可以保证物品只放入一次呢?</p>
<p>倒序就是先算dp</p>
<ul>
<li>
<p>dp = dp] + value = 15 (dp数组已经都初始化为0)</p>
</li>
<li>
<p>dp = dp] + value = 15</p>
</li>
</ul>
<p>所以从后往前循环,每次取得状态不会和之前取得状态重合,这样每种物品就只取一次了。</p>
<p><strong>那么问题又来了,为什么二维dp数组遍历的时候不用倒序呢?</strong></p>
<p>因为对于二维dp,dp都是通过上一层即dp计算而来,本层的dp并不会被覆盖!</p>
<p><strong>再来看看两个嵌套for循环的顺序,代码中是先遍历物品嵌套遍历背包容量,那可不可以先遍历背包容量嵌套遍历物品呢?</strong></p>
<p>不可以!</p>
<p>因为一维dp的写法,背包容量一定是要倒序遍历(原因上面已经讲了),如果遍历背包容量放在上一层,那么每个dp就只会放入一个物品,即:背包里只放入了一个物品。</p>
<p>倒序遍历的原因是,本质上还是一个对二维数组的遍历,并且右下角的值依赖上一层左上角的值,因此需要保证左边的值仍然是上一层的,从右向左覆盖。</p>
<ol start="5">
<li>举例推导dp数组</li>
</ol>
<p>一维dp,分别用物品0,物品1,物品2 来遍历背包,最终得到结果如下:</p>
<p><img src="https://seven97-blog.oss-cn-hangzhou.aliyuncs.com/imgs/202404271524585.png" alt="" loading="lazy"></p>
<h2 id="完全背包问题">完全背包问题</h2>
<p>题目描述:有N件物品和一个最多能背重量为W的背包。第i件物品的重量是weight,得到的价值是value 。<strong>每件物品都有无限个(也就是可以放入背包多次)</strong>,求解将哪些物品装入背包里物品价值总和最大。</p>
<p><strong>完全背包和01背包问题唯一不同的地方就是,每种物品有无限件</strong>。</p>
<p>例子同上,但每个物品有无数个,其实也就是可以重复取同一个物品。问背包能背的物品最大价值是多少?</p>
<p>我们知道 01背包内嵌的循环是从大到小遍历,为了保证每个物品仅被添加一次。</p>
<p>而完全背包的物品是可以添加多次的,所以要从小到大去遍历,即:</p>
<pre><code class="language-java">// 先遍历物品,再遍历背包
for(int i = 0; i < weight.size(); i++) { // 遍历物品
for(int j = weight; j <= bagWeight ; j++) { // 遍历背包容量
dp = max(dp, dp] + value);
}
}
</code></pre>
<p>但是 <strong>为什么遍历物品在外层循环,遍历背包容量在内层循环?</strong>难道就不能遍历背包容量在外层,遍历物品在内层?</p>
<p>01背包中二维dp数组的两个for遍历的先后循序是可以颠倒了,一维dp数组的两个for循环先后循序一定是先遍历物品,再遍历背包容量。</p>
<p><strong>在完全背包中,对于一维dp数组来说,其实两个for循环嵌套顺序是无所谓的!</strong></p>
<p>因为dp 是根据 下标j之前所对应的dp计算出来的。 只要保证下标j之前的dp都是经过计算的就可以了。</p>
<p>遍历物品在外层循环,遍历背包容量在内层循环,状态如图:</p>
<p><img src="https://seven97-blog.oss-cn-hangzhou.aliyuncs.com/imgs/202404271528379.png" alt="" loading="lazy"></p>
<p>遍历背包容量在外层循环,遍历物品在内层循环,状态如图:</p>
<p><img src="https://seven97-blog.oss-cn-hangzhou.aliyuncs.com/imgs/202404271528597.png" alt="" loading="lazy"></p>
<p>看了这两个图,大家就会理解,完全背包中,两个for循环的先后循序,都不影响计算dp所需要的值(这个值就是下标j之前所对应的dp)。</p>
<p>先遍历背包再遍历物品,代码如下:</p>
<pre><code class="language-java">// 先遍历背包,再遍历物品
for(int j = 0; j <= bagWeight; j++) { // 遍历背包容量
for(int i = 0; i < weight.size(); i++) { // 遍历物品
if (j - weight >= 0)
dp = max(dp, dp] + value);
}
}
</code></pre>
<h2 id="多重背包问题">多重背包问题</h2>
<p>题目描述:有N种物品和一个容量为V 的背包。第i种物品最多有Mi件可用,每件耗费的空间是Ci ,价值是Wi 。求解将哪些物品装入背包可使这些物品的耗费的空间 总和不超过背包容量,且价值总和最大。</p>
<p>多重背包和01背包是非常像的, 为什么和01背包像呢?每件物品最多有Mi件可用,把Mi件摊开,其实就是一个01背包问题了。</p>
<p>例如:</p>
<p>背包最大重量为10。</p>
<p>物品为:</p>
<table>
<thead>
<tr>
<th></th>
<th>重量</th>
<th>价值</th>
<th>数量</th>
</tr>
</thead>
<tbody>
<tr>
<td>物品0</td>
<td>1</td>
<td>15</td>
<td>2</td>
</tr>
<tr>
<td>物品1</td>
<td>3</td>
<td>20</td>
<td>3</td>
</tr>
<tr>
<td>物品2</td>
<td>4</td>
<td>30</td>
<td>2</td>
</tr>
</tbody>
</table>
<p>问背包能背的物品最大价值是多少?</p>
<p>和如下情况有区别么?</p>
<table>
<thead>
<tr>
<th></th>
<th>重量</th>
<th>价值</th>
<th>数量</th>
</tr>
</thead>
<tbody>
<tr>
<td>物品0</td>
<td>1</td>
<td>15</td>
<td>1</td>
</tr>
<tr>
<td>物品0</td>
<td>1</td>
<td>15</td>
<td>1</td>
</tr>
<tr>
<td>物品1</td>
<td>3</td>
<td>20</td>
<td>1</td>
</tr>
<tr>
<td>物品1</td>
<td>3</td>
<td>20</td>
<td>1</td>
</tr>
<tr>
<td>物品1</td>
<td>3</td>
<td>20</td>
<td>1</td>
</tr>
<tr>
<td>物品2</td>
<td>4</td>
<td>30</td>
<td>1</td>
</tr>
<tr>
<td>物品2</td>
<td>4</td>
<td>30</td>
<td>1</td>
</tr>
</tbody>
</table>
<p>毫无区别,这就转成了一个01背包问题了,且每个物品只用一次。</p>
<p>代码如下:</p>
<pre><code class="language-java">int[] dp = new int;
//先遍历物品再遍历背包,作为01背包处理
for (int i = 0; i < n; i++) {
for (int j = bagWeight; j >= weight; j--) {
//遍历每种物品的个数
for (int k = 1; k <= nums && (j - k * weight) >= 0; k++) {
dp = Math.max(dp, dp] + k * value);
}
}
}
</code></pre>
</div>
<div id="MySignature" role="contentinfo">
<p>本文来自在线网站:seven的菜鸟成长之路,作者:seven,转载请注明原文链接:www.seven97.top</p><br><br>
来源:https://www.cnblogs.com/sevencoding/p/19376840
頁:
[1]