错题重错之枪战Maf
题目描述
有 n 个人,用1∼n 进行编号,每个人手里有一把手枪。一开始所有人都选定一个人瞄准(有可能瞄准自己)。然后他们按某个顺序开枪,且任意时刻只有一个人开枪。因此,对于不同的开枪顺序,最后死的人也不同。
Input
输入
人数 n<1000000
接下来 n 个数,依次为每个人的 aim
Output
输出只有一行,共两个数,为要求最后死亡数目的最小和最大可能。
Sample Input
8
2 3 2 2 6 7 8 5
Sample Output
3 5
分析
如果\(a\)瞄准\(b\),那么我们就从\(a\)到\(b\)建一条有向边
我们设最大存活程度为\(Max\),最小存活程度为\(Min\)
不难发现,入度为\(0\)的点必定存活(左图中的绿圈)
因为没有人会瞄准他
进一步分析可以发现,建好图后的联通块有两种情况
1、左图(非环)
显然,\(1\)号节点、\(4\)号节点必定存活,而他们所指的\(2\)号节点必死
问题就在\(3\)号节点,如果他先射杀\(2\)号节点,那么他的入度变为\(0\),存活人数为\(3\)
如果\(2\)号节点先射杀\(3\),那么三号节点的入度仍为\(1\),存活人数为\(2\)
所以,我们可以向拓扑排序那样维护入度为\(0\)的点,将其放入队列中
队首的目标必死,队首目标的目标可死可不死,打上标记
2、右图(环状)
最小存活:+1(无论如何都会剩一个人)
最大存活:隔一个人打一个,可以最大存活 len/2;
要特别判断自己打自己的环
代码
#include<stdio.h>
#include<queue>
#include<algorithm>
using namespace std;
const int maxn=1e6+5;
int n,Max,Min,q[maxn],aim[maxn],rd[maxn];
bool die[maxn],undie[maxn];
void Init(){
scanf("%d",&n);
for(int i=1;i<=n;++i){
scanf("%d",&aim[i]);
rd[aim[i]]++;
}
for(int i=1;i<=n;++i)//入度为0的点必活
if(rd[i]==0){
Min++;//最小必活数+1
q[++Max]=i;//队列维护的是入度为0的点
}
}
void Solve(){
int head=1;
while(head<=Max){//扫描队列
int cur=q[head];head++;//取出队首并出队
if(die[aim[cur]]) continue;//队首目标已死(多个入度为0的点指向一个点)
die[aim[cur]]=1;//队首目标必死
int live=aim[aim[cur]];//队首目标的目标可死可不死,看决策
rd[live]--;//
undie[live]=1;//可以让他活,但想死的时候随时可以让他死,在环可以最后死
if(rd[live]==0)//虽然入度为0,但不一定是必活,所以Min不加
q[++Max]=live;
}
//下面处理只剩下环
for(int i=1;i<=n;++i)
if(rd[i] && !die[i]){
int len=0,flag=0;//len记录环大小,flag标记环上是否有undie的点
for(int j=i;!die[j];j=aim[j]){
len++;
flag|=undie[j];
die[j]=1;//标记已死,避免其他的再来计算
}
if(!flag && len>1) Min++;//len=1表示自环,必死
Max+=len/2;
}
printf("%d %d",n-Max,n-Min);
}
int main(){
Init();
Solve();
return 0;
}
错题重错之枪战Maf的更多相关文章
- Java 整体测试重点题 错题积累
重点题 错题积累 1: 解析: %d:用来设置输出日志的日期和时间 %m:用来输出代码中指定的消息 %n:用来输出一个回车换行符 %l:用来输出日志事件的发生位置 %p:用来输出优先级 %f:用 ...
- Ajax与JqueryUI和EasyUI错题总结
Ajax与JqueryUI和EasyUI错题总结 正确答案A,解析:此题考察的是JQuery UI下的menu插件的使用,menu提供ui-state-disabled class 方法禁用任何菜单项 ...
- 错题集锦(二) -- Java专项
错题集锦(二) -- Java专项 标签(空格分隔): 找工作 JVM的内存模型 线程共享: 堆(Heap):主要存放一些对象实例 方法区(Method Area / Non-Heap):用于存储已被 ...
- objective-c 错题
//1, NSString *name = [[NSString alloc]initWithString:@"张三"]; NSLog(@"%d",[name ...
- hibernate错题解析
01 Hibernate错题分析 解析: 此题目考查的是对Hibernate中交叉连接的理解.HQL支持SQL风格的交叉连接查询,交叉连接适用于两个类之间没有定义任何关联时.在where字句中,通 ...
- 20175316盛茂淞 《java程序设计》第三周课堂测试错题总结
20175316盛茂淞 <java程序设计>第三周课堂测试错题总结 出现问题 错题总结 题目1 在Ubuntu中用自己的有位学号建一个文件,教材p87 Example4_15 1. 修改代 ...
- NOIp2014提高组初赛错题简析
总体分析 \(89pts\),粗略来看选择题错的比较多,\(-6pts\).同时又是尿性的填空杀扣了\(5pts\). 不过后面的两大题全对了还是可喜可贺 错题精析 单项选择T8 编译器的主要功能是( ...
- Java多线程习题 ===重点 ,错题积累
多线程重点,错题分析 1: 2: 3: 4: 5: 6: 7: 8: 9: 10: . 12: 13: 14: 15:
- S2 深入.NET和C#编程 笔试测试错题积累
---恢复内容开始--- <深入.NET平台和C#编程>内部测试题-笔试试卷错题积累 1: 1) 以下关于序列化和反序列化的描述错误的是( C). a) 序列化是将对象的状态存储到特定存储 ...
随机推荐
- Servlet 执行时一般实现哪几个方法?
public void init(ServletConfig config): public ServletConfig getServletConfig(): public String getSe ...
- 轻量级进度条 – Nprogress.js
进度条库是前端中常见的库之一,bootstrap中提供了多种进度条样式.NProgress.js和nanobar.js是两款轻量级的进度条组件,使用简便. 官网: NProgress.js:http: ...
- <WP8开发学习笔记>ApplicationBar(任务栏)的切换以及“黑条问题”
ApplicationBar(以下简称AppBar)是WP应用相当常见的控件,也很方便.常见的做法是pivot或者panorama的页面切换的时候,AppBar跟随切换对应的按钮或者不显示按钮,如下图 ...
- @codeforces - 594E@ Cutting the Line
目录 @description@ @solution@ @accepted code@ @details@ @description@ 给定一个字符串 s 与正整数 k.现在你需要进行恰好一次操作: ...
- 2、Redis如何配置成一个windows服务并且设置一键安装卸载与启停
每天启动redis虽然只是一个命令行的事情,但是还是比较烦,所以…… 参考文档:Windows Service Documentation.docx 默认前提:Redis已安装并配置完成(不知道如何配 ...
- 研华advantech-凌华ADLINK板卡运动控制卡
研华advantech:6路独立D/A输出12位分辨率双缓冲D/A转换器多种电压范围:+/-10V,+/-5V,0—+5V,0—+10V和4—20mA电流环(汇)16路数字量输入及16路数字量输出 P ...
- MySQL语句的使用
进入数据库 mysql -u root -pmysql (u用户名,p密码)#如果不想让其他人看到就直接一个p然后回车再打密码 select version(); 查看数据库版本 sele ...
- c语言"##"的使用
#include<stdio.h> #define Operations(x) operation_ ## x // ## 是黏贴字符串 int Operations(sum)(int x ...
- 从Spring Initializr开始
出识springcloud我们这里需要建立两个项目 来感受下微服务 一.配置服务 1. Spring Initializr. 用idea自带的 Spring Initializr. 建立第一个项目 2 ...
- jmeter使用小结(一)
jmeter是用来做接口压力测试的工具.这里只是简单介绍一下使用,大家可以自行查看帮助文档, 1.打开jmeter工具,创建线程组任务 2.添加配置元件,根据需要选择设置 3.添加采样器,这里是htt ...