codeforces3B - Lorry

题面在这里

题意:

有一些物品,体积为1或2,每个物品有一个价值。
现在总共只能放体积为m的物品,问最大的价值。

做法:

显然是贪心。
先说两个正常一点的策略。
1.你先能取多少2就取多少2,然后再把一些价值大一些的1两个一组和2换。
2.直接枚举取多少个2计算一下。

然后来说我的策略QAQ。
首先按照单位价值排序,即你把体积为1的价值×2,然后按照新的价值排序。
然后你就顺序往下取,直到所有物品取完或者容量不够的时候停止。
最后你可能会面临两种情况:
1.容量为0,那你就不用做什么事了。
2.容量为1,这个时候你要把价值最小的1和之前没取的价值最大的2交换一下,使得它的容量刚好用满。
但是这样是错的。
排序的时候应该要双关键字,第二关键字是按照体积从大到小,即2的放前面。
可以看这样一个例子。
3 2
1 10
2 20
1 9
假如没有第二关键字,可能就排成上面这种情况,取的时候肯定是先取1 10,然后容量只剩了1,所以下一个只能取1 9.
但是更优的策略是2 20.
所以可以取2的时候要先取2.

这个方法应该也是正确的(cf上32组数据过了我就不信会有锅(不过可能真的会= =
可能有点复杂

代码:

/*************************************************************
    Problem: codeforces 3B - Lorry
    User: bestFy
    Language: C++
    Result: Accepted
    Time: 156 ms
    Memory: 3600 KB
    Submit_Time: 2018-01-10 11:03:11
*************************************************************/

#include<cstdio>
#include<cstring>
#include<iostream>
#include<cstdlib>
#include<algorithm>
#include<cctype>
#include<cmath>
using namespace std;
typedef long long LL;

const int N = 100010;
int n, m, tot, tmp[N];
struct Node{
    int a, b, id;
}s[N];

const bool cmp(const Node &x, const Node &y) { return x.b > y.b || x.b == y.b && x.a > y.a; }
const bool cmpid(const Node &x, const Node &y) { return x.id < y.id; }
int main()
{
    scanf("%d%d", &n, &m);
    for(int i = 1; i <= n; i ++) {
        scanf("%d%d", &s[i].a, &s[i].b);
        if(s[i].a == 1) s[i].b *= 2;
        s[i].id = i;
    }
    sort(s+1, s+1+n, cmp);
    LL ans = 0; int xxx = 0, iddd;
    for(int i = 1; i <= n; i ++) {
        if(s[i].a == 2 && m == 1) { if(!xxx) xxx = s[i].b, iddd = s[i].id; continue; }
        if(s[i].a == 1) ans += s[i].b/2;
        else ans += s[i].b;
        tmp[++ tot] = s[i].id;
        m -= s[i].a;
        if(!m) break;
    }
    sort(s+1, s+1+n, cmpid);
    if(m) {
        for(int i = tot; i >= 1; i --) if(s[tmp[i]].a == 1) {
            int xx = ans - s[tmp[i]].b/2 + xxx;
            if(xx > ans) ans = xx, tmp[i] = iddd;
            break;
        }
    }
    cout << ans << endl;
    sort(tmp+1, tmp+1+tot);
    for(int i = 1; i <= tot; i ++) printf("%d ", tmp[i]);
    return 0;
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值