UVALive 7146 (贪心+少许数据结构基础)2014acm/icpc区域赛上海站
这是2014年上海区域赛的一道水题。请原谅我现在才发出来,因为我是在太懒了。当然,主要原因是我刚刚做出来。
其实去年我就已经看到这道题了,因为我参加的就是那一场。但是当时我们爆零,伤心的我就再也没有看过那一场的题了。昨天我的队友的高中同学建议我们一起来打一打这场比赛吧,然后我才再次回顾这场比赛。结果一堆琐事,我一共也没有做多久的题,我的队友扎扎实实看了5个小时的题,把另一道水题给过了。全场我们也只过了那么一道题。学姐说,做重现赛和现场赛比较,需要去掉一题,那么我们又爆零了。
题意:
我方有n个人,敌方有m个人。每个人有攻击力a,防御力d。如果我方的人i攻击力比敌方的人j的防御力高,那么i可以杀死j,否则杀不死。当然,如果同时j的攻击力比i个防御力高,那么j同时会杀死i。
但是,每个人只能打一次,无论存活下来还是死去。
求我方在杀死所有敌人的情况下,最多可以留下多少人?
输入:
首行一个整型t,表示共有t组数据。
接下来,每组数据首行两个整型n,m,分别表示我方人数和敌方人数。
接下来n行,每行两个整型a,d,表示我方某人的攻击力与防守力。
接下来m行,每行两个整型a,d,表示敌方某人的攻击力与防守力。
输出:
首先输出”Case #x: ”,x是数据组数。
如果无法杀死敌方所有人,那么输出-1。否则输出我方幸存人数。
题解:
去年的时候这道题我是一点思路也没有,今年有了一点思路,然后顺利的tle了好几发。实际上我距离正确结果只差一步了,但是这一步就是咫尺天涯。
1. 输入的时候我把敌人的攻击和防守反转一下输入,己方的正常输入。
2. 我要把所有人都放在一起,按照攻击a从大到小排序。初始化ans = n;
3. 所有人从大到小读取,直到读取到一个敌人为止。这样,我可以保证,我所有的读取到的我方的人都可以杀死当前的敌人,因为敌人的防守力参与了我方的攻击力的排序。
4. 我要迅速的从我方已经读取的人中寻找一个既可以杀死敌人,又尽可能保存自己的人。即,将我方以读取的人按照防御从小到大排序,找到一个恰好可以防住当前敌人的人。然后将这个人从读取的己方人中删除。表示他杀掉了一个敌人,完成使命。当我方所有人都无法防住敌人的攻击,那就选一个防守最低的人来杀敌,同时也被杀,ans--; 继续读取数组。即,进行步骤3。
5. 当所有的敌人都被杀死时,输出结果。当遇到敌人但无人能杀掉他时,中断读取,输出-1。
写到这里可能有人会有疑问了——
- 为什么在寻找人时可以重新排序(将排好的序打乱)呢?
因为我们只需要保证我们的人能够杀死敌人当前的人就行了。
所以,我们先按照我方的攻击从大到小找人,敌人的防守从大到小找人。这样,我们已经找到的我方的人只要能够杀死当前的敌人,那么就肯定能杀死后面的敌人。那么我们找到的我方的人无论怎么打乱顺序都无所谓了,反正肯定都能杀死敌人。此时的攻击力的顺序已经没有意义了。
- 为什么从可以杀死敌人的我方人中选取刚好能够防住的,如果防不住则选取防最低的,而不直接选取防守最高的呢?
因为,我们是按照我方攻击和敌方防守排的序,所以,无法保证后面防守低的敌人攻击也低。如果我们当前有两个人——
攻击为10,防守为10;
攻击为10,防守为5;
先找到一个敌人——攻击为1,防守为6。那么如果用攻击为10,防守为10的我方士兵。那么如果下一个敌人是攻击为6,防守为1呢?如果用攻击为20,防守为5的,则我方会死一个人。而反过来就都能存活。
如果敌人第一个攻击为11,防守为6,那么如果先用防守高的,则我方会死一个人 下一个敌人攻击为6,防守为6,我方又会死一个人。而反过来只会死一个人。
在保证能够杀死敌人的情况下,优先使用防守低的人,保留防守高的人,能活则活,不能活的就给个死的最痛快的。
然后我就如何将我方的人按照防守力进行第二次排序进行了“深入细致”的思考,结果得到的结论是——使用最快的排序!即k*log2(k)!k表示当前满足条件的我方的人数。哦,我真是SB了!这样一来,我进行了m次排序,且排序长度最差情况是k == n一直到k == n-m+1。可以达到n*n*log2(n)。我不超时谁超时?
后来看了题解,发现,可以用set来保存啊,这样就不需要每次都排一次序了,添加我方第k个人时时间复杂度是log2(k),很稳定,小于n*log2(n)。
我真是脑残了。明明都想到了可以如果把排序降到log2(n)的话一定能过,为什么就是想不到set每次操作是log2(n)呢?涨涨记性,嗯,一定要涨涨记性。
当然,这道题因为会出现重复数据,那么需要用multiset,当然,使用方法大同小异。
还有一点,学会了使用stl的upper_bound()函数。返回值和参数都是一个当前容器的格式。
ps. 当然代码和思路还有些区别,但区别不大。
pss. 头一次写这么长的题解,赶上总结了……
上代码——
#include <cstdio>
#include <cmath>
#include <algorithm>
#include <set>
using namespace std; const int N = ; struct Node
{
int a, d;
int id;
}s1[N], s2[N]; int n, m, t;
int ans; bool cmp(Node x, Node y)
{
return x.a > y.a;
} void Init()
{
scanf("%d%d", &n, &m);
for(int i = ; i < n; i++)
{
scanf("%d%d", &s1[i].a, &s1[i].d);
s1[i].id = ;
}
for(int i = ; i < m; i++)
{
scanf("%d%d", &s2[i].d, &s2[i].a);
s2[i].id = ;
}
sort(s1, s1+n, cmp);
sort(s2, s2+m, cmp);
ans = n;
} void Work()
{
multiset <int> st;
st.clear();
for(int i = , j = ; j < m; j++)
{
while(s2[j].a <= s1[i].a && i < n)
{
st.insert(s1[i].d);
i++;
}
if(st.empty()) {ans = -; return;}
multiset<int>::iterator mid = st.upper_bound(s2[j].d);
if(mid == st.end())
{
st.erase(st.begin());
ans--;
}
else st.erase(mid);
}
} int main()
{
//freopen("test.in", "r", stdin);
scanf("%d", &t);
for(int tm = ; tm <= t; tm++)
{
Init();
Work();
printf("Case #%d: %d\n", tm, ans);
}
}
UVALive 7146 (贪心+少许数据结构基础)2014acm/icpc区域赛上海站的更多相关文章
- 近几年ACM/ICPC区域赛铜牌题
2013 changsha zoj 3726 3728 3736 3735 2013 chengdu hud 4786 4788 4790 2013 hangzhou hdu 4770 4771 47 ...
- 2019年icpc区域赛银川站总结
目录 一.前言 二.10月19日热身赛 三.10月20日正式赛 四.结果 一.前言 比赛前我们队有ccpc厦门和icpc银川的名额,然而这两个地区的时间正好撞了,考虑到银川更容易拿奖,加上我们ACM协 ...
- 2017 ICPC区域赛(西安站)--- J题 LOL(DP)
题目链接 problem description 5 friends play LOL together . Every one should BAN one character and PICK o ...
- 2018 ICPC 区域赛 焦作场 D. Keiichi Tsuchiya the Drift King(计算几何)
http://codeforces.com/gym/102028/problem/D 题意:根据题中给的那个图,然后题目给你 a,b,r,d,让你求出最小的满足矩形通过弯道的w 思路:
- 2018 青岛ICPC区域赛E ZOJ 4062 Plants vs. Zombie(二分答案)
Plants vs. Zombies Time Limit: 2 Seconds Memory Limit: 65536 KB BaoBao and DreamGrid are playin ...
- 2014acm亚洲区域赛陕西赛总结
这次是第一次出来到外面比赛,一切都是非常新鲜的,带着新鲜来到了古城西安.首先感觉就是志愿者一点都不热情.一副爱理不理的,这不是有违我大西北人的热情好客么. 直接说比赛吧. 第一天热身赛,出了两道非常水 ...
- 2016 ACM/ICPC 区域赛(北京) E 题 bfs
https://vjudge.net/problem/UVALive-7672 题意 输入一个五位数n 问由12345变到n的操作最少次数 不可达输出-1 有三种操作 1.交换相邻的位置 次数不 ...
- 算法与数据结构基础 - 贪心(Greedy)
贪心基础 贪心(Greedy)常用于解决最优问题,以期通过某种策略获得一系列局部最优解.从而求得整体最优解. 贪心从局部最优角度考虑,只适用于具备无后效性的问题,即某个状态以前的过程不影响以后的状态. ...
- 算法与数据结构基础 - 堆(Heap)和优先级队列(Priority queue)
堆基础 堆(Heap)是具有这样性质的数据结构:1/完全二叉树 2/所有节点的值大于等于(或小于等于)子节点的值: 图片来源:这里 堆可以用数组存储,插入.删除会触发节点shift_down.shif ...
随机推荐
- nodemon:让node自动重启
nodemon:服务器自动重启工具 当我们修改代码时,node必须要手动重启,但可以按照nodemon. npm install -g nodemon 安装完 nodemon 后,就可以用 nodem ...
- 基于 Express 搭建一个node项目 - 起步
一,如何基于 Express 搭建一个node项目 什么是Express 借用官方的介绍,Express是一个基于Node.js平台的极简.灵活的web应用开发框架,它提供了一系列强大的特性,帮助你创 ...
- influxdb简单使用
之前对influxdb有一个简单的了解和入门的使用,近期由于想使用influxdb做一点东西玩玩,又要捡起influxdb.本篇就针对influxdb的数据库.表的概念,增删改查操作.RESTful操 ...
- bzoj千题计划223:bzoj2816: [ZJOI2012]网络
http://www.lydsy.com/JudgeOnline/problem.php?id=2816 每种颜色搞一个LCT 判断u v之间有边直接相连: 如果u和v之间有边相连,那么他们的深度相差 ...
- ftp服务部署
注:Centos7环境,添加用户指定目录后默认其为此用户的共享目录. chroot_local_user=YES chroot_list_enable=YES # (default follows) ...
- JAVA io 包小结
IO 无非读写 I --> Reader O--> Writer 为了方便字符 或者 文本文件的 操作创造出了 字符流 尤其是 缓冲字符输入输出流(BufferedReader,Buff ...
- github 远程仓库
因为本地Git仓库和GitHub仓库之间的传输是通过SSH加密的,所以设置一下 第1步:创建SSH Key.在用户主目录下,看看有没有.ssh目录,如果有,再看看这个目录下有没有id_rsa和id_r ...
- 基于canvas将图片转化成字符画
字符画大家一定非常熟悉了,那么如何把一张现有的图片转成字符画呢?HTML5让这个可能变成了现实,通过canvas,可以很轻松实现这个功能.其实原理很简单:扫描图片相应位置的像素点,再计算出其灰度值,根 ...
- LCA 算法(一)ST表
介绍一种解决最近公共祖先的在线算法,st表,它是建立在线性中的rmq问题之上. 代码: //LCA: DFS+ST(RMQ) #include<cstdio> #include&l ...
- 关于webpack下热更新?&自动刷新?的小记(非vue-cli)
写本随笔时:webpack4.6.0 为何标题用?号,因为老衲也不知是否用词正确,大概是这样的说法: webpack4.0引入生产模式和开发模式,在开发时使用 webpack 打包后不压缩,所以只需要 ...