Proud Merchants (01背包+ 无后效性)

题目大意:

给你一些钱 m ,然后在这个国家买东西, 共有 n 件物品,每件物品有  价格 P    价值 V    还有一个很特别的属性 Q, Q 指 你如过想买这件物品 你的手中至少有这钱Q 。 虽然你只要花费 钱P ,但你的手中至少有钱Q,如果不足Q ,不能买。问给你钱M ,列出N件物品,最多能获得多少价值的东西

 

分析:

如果没有Q这个限制条件的话,那么这道题就是一个裸的 01背包的问题

好多题解上说按照p-q从大到小排序即可,但是至于为什么并没有说透,所以今天总结一下。

一个问题无论是DP,还是贪心求解,需要满足一个无后效性,其中所谓的无后效性是指:“下一时刻的状态只与当前状态有关,而和当前状态之前的状态无关,当前的状态是对以往决策的总结”。就是说我在当前状态做转移的时候,前面的所有的状态都已经枚举过了,只不过只保留下来了最优解,但是抵达当前状态路径不唯一,仅是最终最优结果唯一

 

不考虑q的限制的01背包的状态转移方程是d[j] = max(d[j],d[j--p[i]]+v[i]);

那么什么时候可以转移呢?j-p[i] 是这个状态的花费,那么m-(j-p[i])是当前我所剩下的钱,只有当m-(j-p[i]) > q[i]的时候,才能转移.

整理一下就p[i] <= j <=m+ p[i] - q[i]; 可以更新的区间是(p[i],m+q[i] - p[i]) 

所谓无后效性是指,前面的在更新时对后面的决策无影响。那么后面的d[j-p[i]]这个状态肯定在前面已经更新到了。所以就是说前面的更新的范围应该比后面大。所以就是p1 - q1 > p2-q2;即按照p-q从大到小排序,或者q - p按照从小到大排序才能保证后面的能更新到不丢失状态。

 

 

#include <iostream>
#include <algorithm>
#include <string>
#include <string.h>
#include <vector>
#include <map>
#include <stack>
#include <set>
#include <queue>
#include <math.h>
#include <cstdio>
#include <iomanip>
#include <time.h>

#define LL long long
#define INF 0x3f3f3f3f
#define ls nod<<1
#define rs (nod<<1)+1

using namespace std;

const int maxn = 1e5 + 10;

struct Node {
    int p,q,v;
}a[510];

bool cmp(Node a,Node b) {
    return a.p - a.q > b.p - b.q;
}

int dp[5010];

int main() {
    int n,m;
    while (~scanf("%d%d",&n,&m)) {
        memset(dp,0, sizeof(dp));
        for (int i = 1;i <= n;i++) {
            scanf("%d%d%d",&a[i].p,&a[i].q,&a[i].v);
        }
        sort(a+1,a+1+n,cmp);
        for (int i = 1;i <= n;i++) {
            for (int j = m;j >= a[i].q;j--)
                dp[j] = max(dp[j],dp[j-a[i].p]+a[i].v);
        }
        printf("%d\n",dp[m]);
    }
    return 0;
}

 

posted @ 2020-02-05 17:20  _Ackerman  阅读(315)  评论(0编辑  收藏  举报