题意:

一个人他有一定的血,有一些怪物,他去杀怪物,有的怪物杀死他后还可以在不费自己血的情况下任意杀死一些怪物,问你他最多杀死多少怪物,在最多杀怪前提下最好用多少血,(大体题意是这样).

思路:

首先我们把怪物分成两个集合,A一个是杀死他后可以免费杀死其他人的,B另一个是杀死他后不能免费杀死其他人的,分析下我们会发现,A集合我们只要杀死其中一个人就可以免费杀死其他人(肯定杀费血最少的),而且很有可能会攒下一些杀死B集合的机会.

其实无非两种情况,杀A集合的和不杀A集合的,如果杀选择杀A集合的那么我们肯定全杀掉,而花费的血就是A集合中血最少的那个,然后我们再把出了刚刚杀的那个以外,和B集合的所有放在一起排序称为C,杀完A集合后会剩下一下免费杀人的机会,我们用这些机会去杀C中的人,既然是免费的肯定从最大的开始杀,杀完后如果没杀没的话我们就用自己的血从小的开始在接着杀,一直杀到自己血没活着敌人没了未知,这是第一种情况,第二种情况是不杀A集合的,只杀B集合的,这种好办,直接把B排序一便,从小的开始能杀多少杀多少,最后在两种方案中选择一个最优的输出来就行了...

/*

1 只杀bi为0的 ,排序下,直接杀就行了;

2 杀bi不为0的 ,先杀死最小的那个bi不为0的,然后得到 s = b1 + b2 + b3 +++

然后在杀死其余的 n1 + n2 - 1 - s 的怪,就ok了; 

*/

#include<stdio.h> 

#include<algorithm>

#define N 100000 + 100

using namespace std;

typedef struct

{

   int xp ,hp;

}NODE;

NODE node1[N] ,node2[N];

bool camp(NODE a ,NODE b)

{

   return a.hp < b.hp;

}

int main ()

{

   int t ,i ,n ,v ,n1 ,n2;

   int ans1_n ,ans2_n ,ans1_hp ,ans2_hp;

   int cas = 1 ,s;

   scanf("%d" ,&t);

   while(t--)

   {

      scanf("%d %d" ,&n ,&v);

      n1 = n2 = s = 0; 

      NODE temp;

      for(i = 1 ;i <= n ;i ++)

      {

         scanf("%d %d" ,&temp.hp ,&temp.xp);

         if(temp.xp)

         {

            node2[++n2] = temp;

            s += temp.xp;

         }

         else

         node1[++n1] = temp;

      } 

      ans1_n = ans1_hp = 0;

      sort(node1 + 1 ,node1 + n1 + 1 ,camp);

      int vv = v;

      for(i = 1 ;i <= n1 ;i ++)

      {

         if(vv < node1[i].hp) break; 

         ans1_n ++;

         ans1_hp += node1[i].hp;

         vv -= node1[i].hp;

      }

         

      sort(node2 + 1 ,node2 + n2 + 1 ,camp);

         

      if(v < node2[1].hp)

      {

         printf("Case %d: %d %d\n" ,cas ++ ,ans1_n ,ans1_hp);

         continue;

      }

         

      ans2_n = 1;

      ans2_hp = node2[1].hp;

      v -= node2[1].hp;

         

      if(s >= n1 + n2 - 1)

      {

         ans2_n = n1 + n2;

         if(ans1_n > ans2_n || ans1_n == ans2_n && ans1_hp < ans2_hp)

         printf("Case %d: %d %d\n" ,cas ++ ,ans1_n ,ans1_hp);

         else

         printf("Case %d: %d %d\n" ,cas ++ ,ans2_n ,ans2_hp);

         continue;

      }

        

      for(i = 2 ;i <= n2 ;i ++)

      node1[++n1] = node2[i];

         

      sort(node1 + 1 ,node1 + n1 + 1 ,camp);

      n1 -= s;

      vv = v;   

      ans2_n += s; 

      for(i = 1 ;i <= n1 ;i ++)

      {

         if(vv < node1[i].hp) break;

         vv -= node1[i].hp;

         ans2_n ++;

         ans2_hp += node1[i].hp;

      }

         

      if(ans1_n > ans2_n || ans1_n == ans2_n && ans1_hp < ans2_hp)

      printf("Case %d: %d %d\n" ,cas ++ ,ans1_n ,ans1_hp);

      else

      printf("Case %d: %d %d\n" ,cas ++ ,ans2_n ,ans2_hp);

   }

   return 0;

}

hdu4415 不错的想法题的更多相关文章

  1. HDU 4972 Bisharp and Charizard 想法题

    Bisharp and Charizard Time Limit: 1 Sec  Memory Limit: 256 MB Description Dragon is watching NBA. He ...

  2. CodeForces 111B - Petya and Divisors 统计..想法题

    找每个数的约数(暴力就够了...1~x^0.5)....看这约数的倍数最后是哪个数...若距离大于了y..统计++...然后将这个约数的最后倍数赋值为当前位置...好叼的想法题.... Program ...

  3. HDU - 5806 NanoApe Loves Sequence Ⅱ 想法题

    http://acm.hdu.edu.cn/showproblem.php?pid=5806 题意:给你一个n元素序列,求第k大的数大于等于m的子序列的个数. 题解:题目要求很奇怪,很多头绪但写不出, ...

  4. hdu 5063 不错的小想法题(逆向处理操作)

    题意:       刚开始的时候给你一个序列,长度为n,分别为a[1]=1,a[2]=2,a[3]=3,a[4]=4...a[n]=n,然后有4种操作如下: Type1: O 1 call fun1( ...

  5. HDU - 5969 最大的位或 想法题

    http://acm.hdu.edu.cn/showproblem.php?pid=5969 (合肥)区域赛签到题...orz 题意:给你l,r,求x|y的max,x,y满足l<=x<=y ...

  6. hdu3006 状态压缩+位运算+hash(小想法题)

    题意:        给了n个集合,问你这n个集合可以组合出多少种集合,可以自己,也可以两个,也可以三个....也可以n个集合组在一起. 思路:       是个小想法题目,要用到二进制压缩,位运算, ...

  7. poj1852 Ants ——想法题、水题

    求最短时间和最长时间. 当两个蚂蚁相遇的时候,可以看做两个蚂蚁穿过,对结果没有影响.O(N)的复杂度 c++版: #include <cstdio> #define min(a, b) ( ...

  8. HDU 4193 Non-negative Partial Sums(想法题,单调队列)

    HDU 4193 题意:给n个数字组成的序列(n <= 10^6).求该序列的循环同构序列中,有多少个序列的随意前i项和均大于或等于0. 思路: 这题看到数据规模认为仅仅能用最多O(nlogn) ...

  9. CodeForces - 156B Suspects 逻辑 线性 想法 题

    题意:有1~N,n(1e5)个嫌疑人,有m个人说真话,每个人的陈述都形如X是凶手,或X不是凶手.现在给出n,m及n个陈述(以+x/-X表示)要求输出每个人说的话是true ,false or notd ...

随机推荐

  1. Java多线程之线程

    前言 线程作为现代操作系统调度的最小单元,多个线程能够同时执行,这将显著提高程序的性能,而且在当前多核CPU的环境下也能更好的利用资源.Java提供了对多线程的良好支持.线程是多线程的基础. 使用多线 ...

  2. git的回滚与撤销【reset and revert】

    git的工作流程-- 3个区域 工作区:我们可以看到的文件内容 在操作 git add 之前的!! 缓存区:是不可见的  已经git add操作,还没git commit -m "" ...

  3. 攻防世界 reverse easy_Maze

    easy_Maze 从题目可得知是简单的迷宫问题 int __cdecl main(int argc, const char **argv, const char **envp) { __int64 ...

  4. 【新阁教育】台达DVP-ES3 ModbusTCP通信案例

    本文主要针对台达DVP-ES3系列PLC,实现上位机与PLC之间的ModbusTCP通信. 一.硬件说明 DVP-ES3 系列为高阶应用可编程控制器, CPU 内置4 组高速计数器输入.4组轴输出(脉 ...

  5. RabbitMQ 入门 (Go) - 1. 简介和安装

    Message Broker(消息代理) 维基百科对 Message Broker 的定义是:Message broker 是一种中介程序模块,它把消息从发送方的正式消息传递协议转化为接收方的正式消息 ...

  6. 亲自动手实现Python+pygame中国象棋游戏

    功能1:实现游戏整体界面显示 一.创建基本的结构 代码如下: import time import pygame def main(): # 初始化pygame pygame.init() # 创建用 ...

  7. Python基础(十六):文件读写,靠这一篇就够了!

    文件读写的流程 类比windows中手动操作txt文档,说明python中如何操作txt文件? 什么是文件的内存对象(文件句柄)? 演示怎么读取文件 ① 演示如下 f = open(r"D: ...

  8. Dynamics CRM9.0安装CRM的时候提示Microsoft.Crm.Setup.Server.RegisterSandboxServiceAction操作失败

    如图: 这种问题有两种情况会遇到.第一种是一个域装多套CRM产品会提示这个问题这个问题是解决不了的. 还一种情况就是安装9.0及以后的产品先安装了ADFS再安装产品就会提示这个错误. 第二种解决办法就 ...

  9. UML相关汇总

    类图 类图是UML最常用的图之一,用于描述面向对象程序设计中,类.接口等结构之间的关系,如图 类图中涉及到以下几种类型的对象 UMLClass 如图中Class1,代表类 UMLOperation 如 ...

  10. OO第四单元总结暨期末总结

    OO第四单元总结暨期末总结 目录 OO第四单元总结暨期末总结 第四单元三次作业架构与迭代 整体感受 HW1 HW2 HW3 四个单元架构设计与方法演进 Unit1 Unit2 Unit3 Unit4 ...