ACM学习历程—HDU4415 Assassin’s Creed(贪心)

Problem Description

Ezio Auditore is a great master as an assassin. Now he has prowled in the enemies’ base successfully. He finds that the only weapon he can use is his cuff sword and the sword has durability m. There are n enemies he wants to kill and killing each enemy needs Ai durability. Every time Ezio kills an enemy he can use the enemy’s sword to kill any other Bi enemies without wasting his cuff sword’s durability. Then the enemy’s sword will break. As a master, Ezio always want to do things perfectly. He decides to kill as many enemies as he can using the minimum durability cost.

Input

The first line contains an integer T, the number of test cases.
For each test case:
The first line contains two integers, above mentioned n and m
(1<=n<=10^5, 1<=m<=10^9).
Next n lines, each line contains two integers Ai, Bi. (0<=Ai<=10^9,
0<=Bi<=10).

Output

For each case, output "Case X: "
(X is the case number starting from 1) followed by the number of the enemies
Ezio can kill and the minimum durability cost.

Sample
Input

2

3 5

4 1

5 1

7 7

2 1

2 2

4 0

Sample
Output

Case 1:
3 4

Case 2: 0 0

题目大意是用m点耐久度去杀人,杀死一个人需要Ai耐久度,但是可以额外再杀死Bi个人。

题目要求的首先是杀人最多,其次是消耗耐久度最少。

首先会想到对人进行分类,一类是B为0的,一类是B不为0的。

因为杀死一个B不为0的,一定能杀死所有B不为0的,而且还能再额外杀死B为0的。

于是就先分出第一种情况:不杀B不为0的人;

这样就是对B为0的人进行贪心,优先杀消耗耐久度最小的即可。

考虑第二种情况:杀死至少一个B不为0的人。

那么首先前提自然是保证能杀死B不为0的人里面消耗耐久度最少的。

然后就可以杀死所有B不为0的人。而且还能再额外杀死B为0的。

最后就是用剩余耐久度去杀死剩下的B为0的人。

但是这样会出现一种情况。就是虽然B不为0的人,可以自动杀死,但是如果我手动杀死一个B不为0的人,可以多出一个名额杀死一个B为0的人。但是也许这样我会消耗更低的耐久度去手动杀死一个人。

于是就发现,肯定可以杀死所有B不为0的人,但是可以优先手动杀死耐久度小的。

得出结论:杀死第一个耐久度最小的B不为0的人后,我便按照耐久度从小到大手动杀人。

此处还要注意的是,我依次杀人后,如果剩下的人都能被自动杀死,我便不需要再手动杀死人了。

最后就是两种情况取最值了。

代码:

#include <iostream>
#include <cstdio>
#include <cstring>
#include <cmath>
#include <algorithm>
#include <set>
#include <utility>
#include <queue>
#include <vector>
#define LL long long

using namespace std;

typedef pair<int, int> pii;
const int maxN = 1e5+5;
int n, m;
struct node
{
    int need;
    int kill;
}p[2][maxN];
int top[2];
int can;

bool cmpNeedLest(node x, node y)
{
    return x.need < y.need;
}

void input()
{
    scanf("%d%d", &n, &m);
    can = 0;
    top[0] = top[1] = 0;
    int need, kill;
    for (int i = 0; i < n; ++i)
    {
        scanf("%d%d", &need, &kill);
        if (kill == 0)
        {
            p[0][top[0]].need = need;
            p[0][top[0]].kill = 0;
            top[0]++;
        }
        else
        {
            p[1][top[1]].need = need;
            p[1][top[1]].kill = kill;
            top[1]++;
            can += kill;
        }
    }
    sort(p[0], p[0]+top[0], cmpNeedLest);
    sort(p[1], p[1]+top[1], cmpNeedLest);
}

pii workOne()
{
    //p->p+top;
    int num = 0, rest = m;
    for (int i = 0; i < top[0]; ++i)
    {
        if (rest >= p[0][i].need)
        {
            num++;
            rest -= p[0][i].need;
        }
        else
            break;
    }
    return pii(num, m-rest);
}

pii workTwo()
{
    int rest = m, num = 0;
    int from[2];
    from[0] = from[1] = 0;
    if (top[1] == 0 || rest < p[1][0].need)
        return pii(0, 0);
    rest -= p[1][0].need;
    from[1]++;
    bool flag;
    for (;;)
    {
        if (n-(from[0]+from[1]) <= can)
            break;
        flag = false;
        if (from[1] < top[1] && p[0][from[0]].need > p[1][from[1]].need)
        {
            if (rest >= p[1][from[1]].need)
            {
                flag = true;
                rest -= p[1][from[1]].need;
                from[1]++;
                num++;
            }
        }
        else if (from[0] < top[0])
        {
            if (rest >= p[0][from[0]].need)
            {
                flag = true;
                rest -= p[0][from[0]].need;
                from[0]++;
                num++;
            }
        }
        if (!flag)
            break;
    }
    return pii(min(from[0]+from[1]+can, n), m-rest);
}

void work()
{
    pii one, two;
    one = workOne();
    two = workTwo();
    if (one.first > two.first)
        printf("%d %d\n", one.first, one.second);
    else if (one.first == two.first)
        printf("%d %d\n", one.first, min(one.second, two.second));
    else
        printf("%d %d\n", two.first, two.second);
}

int main()
{
    //freopen("test.in", "r", stdin);
    int T;
    scanf("%d", &T);
    for (int times = 1; times <= T; ++times)
    {
        input();
        printf("Case %d: ", times);
        work();
    }
    return 0;
}
时间: 2024-12-16 09:58:31

ACM学习历程—HDU4415 Assassin’s Creed(贪心)的相关文章

ACM学习历程——NOJ1113 Game I(贪心 || 线段树)

Description 尼克发明了这样一个游戏:在一个坐标轴上,有一些圆,这些圆的圆心都在x轴上,现在给定一个x轴上的点,保证该点没有在这些圆内(以及圆上),尼克可以以这个点为圆心做任意大小的圆,他想知道自己做多可以与多少个给定的圆相交(相切也算,包含不算). Input 输入有多组数据 输入到文件尾 每一组数据有一个整数n(1<=n<=100000),表示总共有n个圆. 接下是n行,每行两个整数xi,ri表示该圆的圆心坐标和半径. 接下来一行为一个整数x,表示尼克选取点的位置. x xi的范

ACM学习历程—HDU 4726 Kia&#39;s Calculation( 贪心&amp;&amp;计数排序)

DescriptionDoctor Ghee is teaching Kia how to calculate the sum of two integers. But Kia is so careless and alway forget to carry a number when the sum of two digits exceeds 9. For example, when she calculates 4567+5789, she will get 9246, and for 12

ACM学习历程—HDU 5023 A Corrupt Mayor&#39;s Performance Art(广州赛区网赛)(线段树)

Problem Description Corrupt governors always find ways to get dirty money. Paint something, then sell the worthless painting at a high price to someone who wants to bribe him/her on an auction, this seemed a safe way for mayor X to make money. Becaus

ACM学习历程—UESTC 1226 Huatuo&#39;s Medicine(数学)(2015CCPC L)

题目链接:http://acm.uestc.edu.cn/#/problem/show/1226 题目就是构造一个对称的串,除了中间的那个只有1个,其余的两边都是对称的两个,自然答案就是2*n-1. 代码: #include <iostream> #include <cstdio> #include <cstdlib> #include <cmath> #include <cstring> #include <algorithm> #

ACM学习历程—HihoCoder1309任务分配(排序 &amp;&amp; 贪心)

http://hihocoder.com/problemset/problem/1309 题目大意是给定n个任务的起始时间,求问最少需要多少台机器. 有一个贪心的策略就是,如果说对于一个任务结束,必然接一个开始时间最接近这个的比较合算.我们假想一个任务池,那么任务池中最早结束的那个,必然接剩余任务中最早开始的比赛合算(相同开始时间最早结束),而且假设这个任务是P,那么对于一个结束时间晚于P的任务,显然后面的一段时间是浪费的,同样对于一个开始时间晚于P的任务,前面一段是浪费的: 关键在于某个开始时

ACM学习历程—FZU 2144 Shooting Game(计算几何 &amp;&amp; 贪心 &amp;&amp; 排序)

Description Fat brother and Maze are playing a kind of special (hentai) game in the playground. (Maybe it’s the OOXX game which decrypted in the last problem, who knows.) But as they don’t like using repellent while playing this kind of special (hent

ACM学习历程——HDU 5014 Number Sequence (贪心)(2014西安网赛)

Description There is a special number sequence which has n+1 integers. For each number in sequence, we have two rules: ● a i ∈ [0,n] ● a i ≠ a j( i ≠ j ) For sequence a and sequence b, the integrating degree t is defined as follows(“?” denotes exclus

ACM学习历程——POJ 2376 Cleaning Shifts(贪心)

Description Farmer John is assigning some of his N (1 <= N <= 25,000) cows to do some cleaning chores around the barn. He always wants to have one cow working on cleaning things up and has divided the day into T shifts (1 <= T <= 1,000,000), t

ACM学习历程——POJ 1700 Crossing River(贪心)

Description A group of N people wishes to go across a river with only one boat, which can at most carry two persons. Therefore some sort of shuttle arrangement must be arranged in order to row the boat back and forth so that all people may cross. Eac