题目链接

https://www.lydsy.com/JudgeOnline/problem.php?id=1124

https://www.luogu.org/problemnew/show/P3472

分析

首先, 每个神枪手都只有一个目标.

若是把每个神枪手当成一个点来建图, 那么这个图每个点的出度都是1(基环内向树)

既然\(N \leq 10^6\), 这道题基本上是个贪心.

分别考虑最少存活人数和最大存活人数的求解. (死亡人数 = N - 存活人数)

1. 最少存活人数(minlive)

若是一个点入度为0, 那么这个点必定存活.

对于基环内向森林中的每一个基环内向树, 判断其中是否有入度为0的点.

若有, 那么这个基环内向树中除了入度为0的点其他点都可以被杀(环上的点先开枪使环上只留1个点, 再按照拓扑序逆序开枪即可)

此时minlive += cnt[ ind==0 ]

若无, 那么其中必定1个点可以存活, minlive++(环长为1的需要特判).

2. 最多存活人数(maxlive)

这个问题相对复杂.

先考虑内向树上maxlive的求解.

内向树上叶子节点必定存活. 而且, 从下到上, 每一层的节点个数都\(\ge\) 其上一层节点个数.所以取从下到上的1, 3, 5, 7....层是最优方案.

如: 这棵内向树有3层.

所以, 将入度为0的点(最下层点集)取出, 用类似于拓扑排序分层的方法, 隔一层向队列中加点即可(详见代码).

具体来说, 如果a杀死了b, 那么b指向的人c就少了一个可以杀死他的人. 所以把c的入度-1. 当c入度为0时, c必定存活, 那么可以把c加入队列, 作为a这一层之后的"活人"层(这一点与拓扑排序相同)

再考虑基环内向树上maxlive的求解.

用以上算法求解过后, 图中必定剩下若干个环.

这些环对maxlive的贡献是 环长/2.

code

#include<bits/stdc++.h>
using namespace std; typedef long long ll;
#define rep(i, j, k) for(register int i=(j);i<=(k);++i)
#define per(i, j, k) for(register int i=(j);i>=(k);--i) int read(){
int ret = 0, f = 1; char c=getchar();
while(isdigit(c) == false) {
if(c=='-')f=-1;
c=getchar();
}
while(isdigit(c)) ret = ret*10+c-'0', c=getchar();
return ret*f;
}
const int maxn= 1e6+5;
int n;
int to[maxn], ind[maxn];
int L, R, q[maxn];
int maxlive, minlive; bool dead[maxn], vis[maxn];
void getdead(){
minlive = 0;
L = R = 1; //[L, R)
rep(i, 1, n){
if(ind[i] == 0){
minlive++;
q[R++] = i; //将入度为0的点加入队列
for(int cur = i; !vis[cur]; vis[cur] = 1, cur = to[cur]);
}
}
rep(i, 1, n) if(!vis[i]){
int looplen = 0;
for(int cur = i; !vis[cur]; vis[cur] = 1, cur = to[cur])looplen++;
if(looplen != 1) {
minlive += 1;
}
} //calculate minlive memset(vis, 0, sizeof vis);
while(R - L > 0){
int cur = q[L++], u = to[cur];
if(vis[cur]) continue;
maxlive++, vis[cur] = 1;//cur 存活
if(vis[u] == 0){ //给to[u]除去一个威胁
vis[u] = 1, ind[to[u]]--;
if(ind[to[u]] == 0) q[R++] = to[u];
}
assert(u <= n);
}
// printf("bfs : %d %d\n" ,maxlive, minlive);
rep(i, 1, n) if(!vis[i]) {
int looplen = 0, cur;
for(cur = i; !vis[cur]; vis[cur] = 1, cur = to[cur])looplen++;
if(cur == i) maxlive += (looplen)/2;
else maxlive += (looplen+1)/2;
} //calculate maxlive
// printf("clearloop : %d %d\n" ,maxlive, minlive); }
signed main(){
// freopen("5.in", "r", stdin);
n =read();
rep(i, 1, n) {
to[i] = read();
ind[to[i]]++;
}
getdead();
printf("%d %d\n", n - maxlive, n - minlive);
return 0;
}

bzoj1124_枪战_基环树的更多相关文章

  1. [bzoj3037/2068]创世纪[Poi2004]SZP_树形dp_并查集_基环树

    创世纪 SZP bzoj-3037/2068 Poi-2004 题目大意:给你n个物品,每个物品可以且仅可以控制一个物品.问:选取一些物品,使得对于任意的一个被选取的物品来讲,都存在一个没有被选取的物 ...

  2. 【LOJ523】[LibreOJ β Round #3]绯色 IOI(悬念)(霍尔定理_基环树)

    题目 LOJ523 官方题解 分析 由于某些原因,以下用「左侧点」和「右侧点」分别代替题目中的「妹子」和「男生」. 根据题意,显然能得出一个左侧点只能向一个或两个右侧点连边.这似乎启发我们把左侧点不看 ...

  3. [bzoj1040][ZJOI2008]骑士_树形dp_基环树_并查集

    骑士 bzoj-1040 ZJOI-2008 题目大意:n个骑士,每个骑士有权值val和一个讨厌的骑士.如果一个骑士讨厌另一个骑士那么他们将不会一起出战.问出战的骑士最大atk是多少. 注释:$1\l ...

  4. Hdu第八场 树形dp+基环树

    Card Game 每个牌背面的数字朝正面的数字连一条有向边 则题目变为问你最少翻转多少次 能使得每个数字的入度不超过1 首先判断图中每个连通块是不是树或者基环树 因为只有树或者基环树能使得每个点的入 ...

  5. 【BZOJ1791】【IOI2008】【基环树】island(status第一速度)

      1791: [Ioi2008]Island 岛屿  Time Limit: 20 Sec  Memory Limit: 162 MB Submit: 908  Solved: 159 [Su ...

  6. BZOJ_4636_蒟蒻的数列_线段树+动态开点

    BZOJ_4636_蒟蒻的数列_线段树+动态开点 Description 蒟蒻DCrusher不仅喜欢玩扑克,还喜欢研究数列 题目描述 DCrusher有一个数列,初始值均为0,他进行N次操作,每次将 ...

  7. BZOJ_4627_[BeiJing2016]回转寿司_离散化+树状数组

    BZOJ_4627_[BeiJing2016]回转寿司_离散化+树状数组 Description 酷爱日料的小Z经常光顾学校东门外的回转寿司店.在这里,一盘盘寿司通过传送带依次呈现在小Z眼前.不同的寿 ...

  8. BZOJ_3252_攻略_线段树+dfs序

    BZOJ_3252_攻略_线段树+dfs序 Description 题目简述:树版[k取方格数] 众所周知,桂木桂马是攻略之神,开启攻略之神模式后,他可以同时攻略k部游戏.今天他得到了一款新游戏< ...

  9. BZOJ_4653_[Noi2016]区间_线段树+离散化+双指针

    BZOJ_4653_[Noi2016]区间_线段树+离散化+双指针 Description 在数轴上有 n个闭区间 [l1,r1],[l2,r2],...,[ln,rn].现在要从中选出 m 个区间, ...

随机推荐

  1. tomcat设置为开机自启动

    第一步:设置环境变量(在java环境变量配置完成的情况下) 计算机右键——>属性——>高级系统设置——>环境变量——>在用户变量中心新建CATALINA_HOME变量 编辑pa ...

  2. JAVA基础---入门

    JDK的安装和环境变量的配置: 在Oralce官网下载好符合自己电脑配置的JDK后开始配置环境变量. 找到下载好的JDK的位置,复制,然后在环境变量里创建“JAVA_HOME”,粘贴:在path里用“ ...

  3. Navicat连接阿里云服务器Linux下的Mysql

    用Navicat连接阿里云ECS服务器上的MySQL数据库   今天用navtive连接阿里云服务器(Linux)的数据库时,老是连接不上,并且报10060错误,要通过以下两个步骤解决: 1.先进入l ...

  4. java连接3种数据库 JdbcLinkDB --201801

    先看这篇记录 java连接3种数据库 JdbcLinkDB 测试 --201801 配置文件放在jar外面 读取,遇到的问题 - 海蓝steven - 博客园https://www.cnblogs.c ...

  5. 论文翻译:BinaryConnect: Training Deep Neural Networks with binary weights during propagations

    目录 摘要 1.引言 2.BinaryConnect 2.1 +1 or -1 2.2确定性与随机性二值化 2.3 Propagations vs updates 2.4 Clipping 2.5 A ...

  6. 在Cyclone IVE中使用进位链的几个规则

    最近在FPGA上做ps级的Delay line,所以认真剖析了一下Cyclone IVE4的布局布线延迟.这里说明CARRY链的几个特性规则,如有错误请各位大大指出,谢谢.(另外由于匆忙没有时间验证其 ...

  7. windows 下 配置 github

       github   功能介绍 1. 记录多个版本 2.查看历史操作,可以进行版本回退和前进的控制 3. 多端共享代码,自动合成  Github  与  SVN   1.  SVN 版本集中管理,所 ...

  8. 移动端rem与px适应js

    方法一: (function (doc, win) { var docEl = doc.documentElement, resizeEvt = "orientationchange&quo ...

  9. tcp与ip协议的区别

    TCP/IP(TransmissionControlProtocol/InternetProtocol的简写,中文译名为传输控制协议/互联网络协议). 简单地说,就是由底层的IP协议和TCP协议组成的 ...

  10. 怎么编辑PDF文件内容,PDF文件编辑方法

    怎样编辑PDF文件内容?这是一个常常困扰我们的问题,工作当中我们经常会收到PDF格式的文件,但有时的文件内容不是我们想要的或者是觉得不合理的需要改掉.但是每次有这样的问题时都没有什么好的解决方法,每次 ...