关于DP与背包

听说过动态规划(DP)的同学应该都知道有背包问题的存在。

首先我们来了解一下动态规划

基本思想:

动态规划算法通常用于求解具有某种最优性质的问题。在这类问题中, 可能会有很多可行解。没一个解都对应于一个值,我们希望找到具有最优值的解。胎动规划算法与分治法类似,其基本思想也是将待求解问题分解为若干个子问题,先求解子问题,然后从这些子问题的解得到原问题的解。与分治法不同的是,适用于动态规划算法求解的问题,经分解得到的子问题往往不是互相独立的。若用分治法来解这类问题,则分解得到的子问题数目太多,有些子问题被重复计算很多次。如果我们能保存已解决子问题的答案,而在需要时再找出已求得的答案,这样就可以避免大量的重复计算,节省时间。我们可以用一个表来记录所有已解决的子问题的答案。不管该子问题以后是否被用到,只要它被计算过,就将其结果填入表中。这就是动态规划算法的基本思路。具体的动态规划算法多种多样,但它们具有相同的填表格式。

与分治法最大的差别是:适用于动态规划求解的问题,经分解后得到的子问题往往不是互相独立的(即下一个子阶段的求解是建立在上一个子阶段的解的基础上,进行进一步的求解)

应用场景:

适用于动态规划的问题必须满足最优化原理、无后效性和重叠性。

(1) 最优化原理(最优子结构性质):一个最优化策略具有这样的性质,不论过去状态和决策如何,对前面的决策所形成的状态而言,余下的决策必须构成最优策略。简而言之,一个最优化策略的子策略总是最优的。一个问题满足最优化原理又称其具有最优子结构性质。

(2) 无后效性:将各阶段按照一定的次序排列好之后,对于某个给定的阶段状态,它以前各阶段的状态无法直接影响它未来的决策,而只能通过当前的这个状态。换句话说,每个状态都是过去历史的一个完整总结。这就是无后向性,又称无后效性。

(3) 子问题的重叠性:动态规划将原来具有指数级时间复杂度的搜索算法改进成了具有多项式时间复杂度的算法。其中的关键在于解决冗余,这就是动态规划算法的根本目的。动态规划实质上是一种以空间换时间的技术,它在实现的过程中,不得不存储产生过程中的各种状态,所以它的空间复杂度要大于其他算法。

接下来,看背包问题,嗯,听起来就很朴素。

其实背包问题也很简单,无非就是在容器一定的情况下,往里面塞东西,但是容器的容量有限,总不能把它撑爆吧,那不就弄巧成拙了,所以我们要把利益和效率最大化。在容器容量一定的情况下,根据物体的体积(质量)等条件来判断如何装才能使容器内所存物体的价值最大。这就用到了背包问题。

首先我们来看的是01背包问题

01背包指在容器容量一定下,每种物品只有一个,告诉你他们的体积(或者其它条件),以及价值,来使价值最大化。

代码:(优化后为二重循环)

#include<cstring>
#include<iostream>
#include<algorithm>
using namespace std;
int total_weight,wp_weight,every_jz;
int main() {
    cin>>total_weight;
    int dp[total_weight];
    memset(dp,0,sizeof(dp));
    cin>>wp_weight;
    int w[wp_weight],v[wp_weight];
    for(int i=1;i<=wp_weight;i++)
        cin>>w[i];
    for(int i=1;i<=wp_weight;i++)
        cin>>v[i];
    for (int i = 1; i <= wp_weight; i++)
        for (int j = total_weight; j >=w[i] ; j--)
            dp[j] = max(dp[j], dp[j - w[i]] + v[i]);
    cout << "总的价值为: " << dp[total_weight] << endl;
    return 0;
}

进一步的,我们再来看一下完全背包问题

完全背包指在容器容量一定下,每种物品有无数件,告诉你他们每种分别的体积(或者其它条件),价值,力求价值最大化。

代码:(优化后为二重循环)

#include<cstring>
#include<iostream>
#include<algorithm>
using namespace std;
int total_weight,wp_weight,every_jz;
int main() {
    cin>>total_weight;
    int dp[total_weight];
    memset(dp,0,sizeof(dp));
    cin>>wp_weight;
    int w[wp_weight],v[wp_weight];
    for(int i=1;i<=wp_weight;i++)
        cin>>w[i];
    for(int i=1;i<=wp_weight;i++)
        cin>>v[i];
    for (int i = 1; i <= wp_weight; i++)
        for (int j = w[i]; j <=total_weight ; j++)
            dp[j] = max(dp[j], dp[j - w[i]] + v[i]);
    cout << "总的价值为: " << dp[total_weight] << endl;
    return 0;
}

最后我们来看一下多重背包问题

即容器容量一定下,每种物品有固定件数,告诉你他们每种分别的体积(或者其它条件),价值,自然还是令价值最大化。

代码:

#include<iostream>
#include<cstring>
#include<algorithm>
using namespace std;
int total_weight,w_weight;
int main() {
    cin>>total_weight>>w_weight;
    int w[w_weight],v[w_weight],cot[w_weight],dp[total_weight+1];
    memset(dp,0,sizeof(dp));
    for (int i = 1; i <= w_weight; i++)
        for (int k = 1; k <= cot[i]; k++)
            for (int j = total_weight; j >= w[i]; j--)
                dp[j] = max(dp[j], dp[j - w[i]] + v[i]);
    cout << "总的价值为: " << dp[total_weight] << endl;
    return 0;
}

别丧气,你现在只是单身了十几年,以后,你还会单身好久呢。

原文地址:https://www.cnblogs.com/GTBD/p/9215470.html

时间: 2024-11-07 16:51:07

关于DP与背包的相关文章

poj1384——dp,完全背包

POJ 1384  dp,完全背包 Piggy-Bank Time Limit: 1000MS   Memory Limit: 10000K Total Submissions: 8404   Accepted: 4082 Description Before ACM can do anything, a budget must be prepared and the necessary financial support obtained. The main income for this a

USACO Money Systems Dp 01背包

一道经典的Dp..01背包 定义dp[i] 为需要构造的数字为i 的所有方法数 一开始的时候是这么想的 for(i = 1; i <= N; ++i){ for(j = 1; j <= V; ++j){ if(i - a[j] > 0){ dp[i] += dp[i - a[j]]; } } } 状态存在冗余, 输出的时候答案肯定不对 但只需要改一下两个for循环的顺序即可. Source Code: /* ID: wushuai2 PROG: money LANG: C++ */ //

hdu4003 树形dp+分组背包

http://acm.hdu.edu.cn/showproblem.php?pid=4003 Problem Description Humans have discovered a kind of new metal mineral on Mars which are distributed in point‐like with paths connecting each of them which formed a tree. Now Humans launches k robots on

hdu 1561The more, The Better(树形dp&amp;01背包)

The more, The Better Time Limit: 6000/2000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) Total Submission(s): 4949    Accepted Submission(s): 2918 Problem Description ACboy很喜欢玩一种战略游戏,在一个地图上,有N座城堡,每座城堡都有一定的宝物,在每次游戏中ACboy允许攻克M个城堡并获得里面的宝

poj1014 dp 多重背包

1 //Accepted 624 KB 16 ms 2 //dp 背包 多重背包 3 #include <cstdio> 4 #include <cstring> 5 #include <iostream> 6 using namespace std; 7 const int imax_n = 120005; 8 int f[imax_n]; 9 int amount[7]; 10 int v; 11 int n=6; 12 int max(int a,int b) 1

poj 3345 Bribing FIPA 【树形dp + 01背包】

Bribing FIPA Time Limit: 2000MS   Memory Limit: 65536K Total Submissions: 4274   Accepted: 1337 Description There is going to be a voting at FIPA (Fédération Internationale de Programmation Association) to determine the host of the next IPWC (Interna

HDU ACM 4044 GeoDefense -&gt;树形DP+分组背包

题意:地图是一个编号为1-n的节点的树,节点1是敌方基地,其他叶节点是我方基地.敌人基地会出来敌人,为了防止敌人攻进我方基地,我们可以选择造塔.每个节点只能造一个塔,节点i有ki种塔供选择,价值和攻击力为price_i, power_i,攻击力power_i是让敌人经过这个节点时让敌人的HP减少power_i点.因此从敌人基地到我方任意一个基地的路径,这条路径上所有塔的攻击力之和,就是这个基地的抵抗力. 敌人攻击路径不确定,为了保护我方所有基地,需要确定所有基地中抵抗力最低的一个.我方只有数量为

hdu 5389 dp类似背包

http://acm.hdu.edu.cn/showproblem.php?pid=5389 Problem Description Zero Escape, is a visual novel adventure video game directed by Kotaro Uchikoshi (you may hear about ever17?) and developed by Chunsoft. Stilwell is enjoying the first chapter of this

poj 1837 Balance (dp,01背包)

链接:poj 1837 题意:有一个天平,天平左右两边各有若干个钩子,总共有C个钩子,有G个钩码, 求将钩码挂到钩子上使天平平衡的方法的总数.其中可以把天枰看做一个以x轴0点作为平衡点的横轴 分析:力臂=重量 *臂长 = g[i]*c[j] 当平衡度k=0时,说明天枰达到平衡,k>0,说明天枰倾向右边(x轴右半轴),k<0则左倾 因此可以定义一个 状态数组dp[i][k],意为在挂满前i个钩码时,平衡度为k的挂法的数量. 由于距离c[i]的范围是-15~15,钩码重量的范围是1~25,钩码数量

ZOJ 3450 Doraemon&#39;s Railgun (DP&#183;分组背包)

题意  多啦A梦有一个超电磁炮  然后要打死n堆敌人  在同一条射线上的敌人只有先打死前面的一堆才能打后面的一堆  给你打死某堆敌人需要的时间和这堆敌人的人数   问你在T0时间内最多打死多少个敌人 分组背包问题  先要把同一条射线上的敌人放到一个分组里  后面的敌人的时间和人数都要加上前面所有的  因为只有前面的都打完了才能打后面的  然后每组最多只能选择一个   判断共线用向量处理   然后去背包就行了 注意给你的样例可能出现t=0的情况   在分组时需要处理一下    被这里卡了好久 #i