|  
  
 
 题目描述:一共有 k种 物品(气缸),对于第i种物品,第一维 费用(氧气体积) 是v1i,第二维 费用(氮气体积) 是v2i,价值(气缸重量) 是wi每个物品只能 被选一次
 求一个 选择方案,使得第一维费用 至少为 n,第二维费用 至少为m且 总价值最小 思路:闫氏dp分析法:
  状态表示dp[i,j,k]:集合:所有从前
 i个物品中选,且氧气含量至少是j,氮气含量至少是k的所有选法属性:所有从前
 i个物品中选,且氧气含量至少是j,氮气含量至少是k的所有选法的气缸重量总和的最小值 状态计算:所有 不包含物品
 i的所有选法:dp[i - 1, j, k]所有 包含物品
 i的所有选法:dp[i - 1, j - ai, k - bi]因此,状态转移方程:
 dp[j, k] = min(dp[j, k], dp[max(0, j - ai), max(0, k - bi)] + ci);(省去第一维) 细节处理:(1)初始化的细节处理:(对于要求体积 “至少是” 问题的初始化处理)
 ①
 dp(0)(0)(0)=0:第一维为0(一件物品都不选),第二维和第三维体积均为0(氧气和氮气体积至少为0),这种情况是合法的,且
 min值为0。②
 dp(0)(1,2,3...)(1,2,3...):第一维为0(一件物品都不选),第二维和第三维体积至少为i(1,2,3…),根据实际,这些状态都是不合法的(无法求
 min),因此将它们初始化为∞即可(至于是+∞还是-∞取决于这个问题是求的最大值还是最小值)
 (2)即使所需要的氧气或者氮气所需的是数量是负数,但其所需数量与
 0是等价的,因此可以通过所需数量为0来转移,举个例子:体积至少为-2, 这种情况是存在的,体积是0,至少是-2,体积是10,至少也是-2。因此:
 j需要遍历到0,k需要遍历到0。 时间复杂度:
    
     
      
       
        O
       
       
        (
       
       
        n
       
       
        m
       
       
        k
       
       
        )
       
      
      
       O(nmk)
      
     
    O(nmk) 代码:#include<bits/stdc++.h>
using namespace std;
int n,m;
int k;
int ai,bi,ci;//第 i 个气缸里的氧和氮的容量及气缸重量
const int N = 22,M = 80;//氧,氮各自需要的量数据范围 
int dp[N][M];
int main()
{
	cin>>m>>n;
	cin>>k;
	//初始化 
	memset(dp,0x3f,sizeof dp);
//其它的所有方案(f[0][1,2,3...]),由于第一维为0(一件物品都不选),所以第二维体积必须为0不能为i(1,2,3...),
//这些状态都是不合法的,因此将它们初始化为∞即可
	dp[0][0]=0;
//第一维为0(一件物品都不选,代码中第一维),第二维体积为0此时只有f[0][0]是合法的
	
	for(int i=0;i<k;++i)
	{
		cin>>ai>>bi>>ci;
		for(int j=m;j>=0;--j)
		{
			for(int k=n;k>=0;--k)
			{
				dp[j][k]=min(dp[j][k],dp[max(0,j-ai)][max(0,k-bi)]+ci);
//这两维体积可以为负,此时置为0即可。举个例子:体积至少为-2,这种情况是存在的,体积是5,至少是-2,体积是10,
//至少也是-2
			}
		}
	}
	cout<<dp[m][n]<<endl;
//所有从前i个物品中选,且氧气含量至少是j, 氮气含量至少是k 的所有选法中 所需的气缸的重量总和的最低值
	return 0;
}
 扩展:求对于三类最大值最小值题型的初始化总结  一、 体积最多是j(对应背包问题模板题) 
    
   ①二维:f[i,k] = 0,0 <= i <= n,0 <= k <= m(只会求价值的最大值)
 
   ②一维:f[i] = 0,0 <= i <= m(只会求价值的最大值)
 
  从实际含义进行推理,f[0,i]表示:一件物品都不选的情况下,体积最多为i时的最大价值,i不管取多少,什么都不选是一种方案,每个f[0,i]至少包含一个方案,f[0,i]的最大值为0。
 二、 体积恰好是j 
  dp(0)=0,dp(1,2,3...) = ∞,并保证V>=0
 
    
  当求价值的最小值:f[0][0] = 0, 其余是inf
 
  当求价值的最大值:f[0][0] = 0, 其余是-inf
 
    
  当求价值的最小值:f[0] = 0, 其余是inf
 
  当求价值的最大值:f[0] = 0, 其余是-inf
 
  此时,就只有f[0][0]是合法的,其它的所有方案(f[0][1,2,3...]),由于一件物品都不选,所以体积必须为0不能为i,这样的状态都是不合法的,因此将它们初始化为∞即可,至于是+∞还是-∞则取决于这个问题是求的最大值还是最小值。
 三、 体积至少为j 
  dp(0)=0,dp(1,2,3...)=∞,无需保证V>=0
 
   ①二维:体积至少j,f[0][0] = 0,其余是inf(只会求价值的最小值)
 
   ②一维:体积至少j,f[0] = 0,其余是inf(只会求价值的最小值)
 
  状态计算与第二种相同,只不过V可以为负,此时置为0即可。
 |