P3472 [POI2008]MAF-Mafia

解法

不难发现,这个题目建图后出现的是一个基环树森林+一堆环,因为每个点仅有一条出边。

大概长这样:

对于最大,最小值,我们分开考虑。

最大值

可以看到,当我们缩点后,对于一个没有入度的点,其对答案的贡献为1。

因为当一个点入度为0时,其要么原本是个环,要么原本也是一个点:

当其是一个环时,可以通过一些击杀顺序从而使环仅剩一个点。

当其是一个点时,没有点可以杀掉他。

而其他点都可以通过一些奇妙的顺序被杀掉。

最小值

我们观察,如果是一个单独的环,其对答案的贡献应该是\(siz/2\)

我们再在一颗基环树上观察:

可以得到,入读为\(0\)的点是不会被杀的,那么我们将其入队,并统计答案。

然后入读为\(0\)的点所指向的点一定会被杀,当我们将队头取出时,将其所指向的点杀掉,再将其指向的点所指向的点的入读减\(1\),当其入度为\(0\)时,便将其入队,并统计答案。

这里就相当于做了一次变形的拓扑排序。

然后我们再去找环:

当这个环上至少有一个点被杀时,那么这个环就变成链,再拓扑排序的时候已经计算过了不管。

如果没有点被杀,则和单独一个环的情况一样。

CODE

#include<bits/stdc++.h>
#define ll long long
using namespace std;
const int maxn = 1e6+33;
inline ll read_int(){
ll a=0;bool f=0;char g=getchar();
while(g<'0'||'9'<g) {if(g=='-') f=1;g=getchar();}
while('0'<=g&&g<='9') a=a*10+g-'0',g=getchar();
return f ? -a : a;
} inline void write(ll a,bool b=1){
if(a<0) a=-a,putchar('-');
int lin[30],top=0;
while(a) lin[++top]=a%10,a/=10;
if(top==0) lin[++top]=0;
while(top) putchar(lin[top--]+'0');
if(b) putchar('\n');
}
int n;
int t[maxn],rd[maxn]; int tme,low[maxn],dfn[maxn],zan[maxn],top;
int bel[maxn],siz[maxn],tot;
int vis[maxn]; inline void tarjian(int s){
low[s]=dfn[s]=++tme;
zan[++top]=s;
if(!vis[t[s]]){
if(dfn[t[s]]) low[s]=min(low[s],dfn[t[s]]);
else tarjian(t[s]),low[s]=min(low[s],low[t[s]]);
}
vis[s]=1;
if(low[s]==dfn[s]){
tot++;
while(zan[top]!=s){
siz[tot]++;
bel[zan[top]]=tot;
top--;
}
siz[tot]++;
bel[zan[top]]=tot;
top--;
}
} int maxx=0,rd_[maxn]; inline void suodian(){
for(int i=1;i<=n;i++){
if(rd[i]==0) maxx++;
if(bel[i]==bel[t[i]]){
if(i==t[i]) rd_[bel[i]]=1;
continue;
}
rd_[bel[t[i]]]++;
}
for(int i=1;i<=tot;i++){
if(rd_[i]) continue;
if(siz[i]==1) continue;
maxx+=1;
}
maxx=n-maxx;
} int die[maxn],dee[maxn];
inline void read(){
n=read_int();
for(int i=1;i<=n;i++) t[i]=read_int(),rd[t[i]]++;
for(int i=1;i<=n;i++) if(vis[i]==0) tarjian(i);
suodian();
queue<int> p;
ll minn=0;
for(int i=1;i<=n;i++){
if(rd[i]) continue;
minn++;
p.push(i);
}
while(!p.empty()){
int s=p.front();
p.pop();
if(die[t[s]]) continue;
die[t[s]]=1;
if((--rd[t[t[s]]])==0) p.push(t[t[s]]),minn++;
}
for(int i=1;i<=n;i++) dee[bel[i]]|=die[i];
for(int i=1;i<=tot;i++) if(dee[i]==0&&siz[i]>1) minn+=siz[i]/2;
cout<<n-minn<<" "<<maxx;
} int main (){
// freopen("maf.in","r",stdin);
// freopen("maf.out","w",stdout);
read();
// while(1) getchar();
}

后记

模拟赛没写出来,木有判断自环。。。。

P3472 [POI2008]MAF-Mafia的更多相关文章

  1. BZOJ 1124: [POI2008]枪战Maf

    1124: [POI2008]枪战Maf Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 617  Solved: 236[Submit][Status ...

  2. [POI2008]枪战Maf

    [POI2008]枪战Maf 题目 有n个人,每个人手里有一把手枪.一开始所有人都选定一个人瞄准(有可能瞄准自己).然后他们按某个顺序开枪,且任意时刻只有一个人开枪.因此,对于不同的开枪顺序,最后死的 ...

  3. bzoj 1124 [POI2008]枪战Maf 贪心

    [POI2008]枪战Maf Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 741  Solved: 295[Submit][Status][Disc ...

  4. 【BZOJ1124】[POI2008]枪战Maf 贪心+思路题

    [BZOJ1124][POI2008]枪战Maf Description 有n个人,每个人手里有一把手枪.一开始所有人都选定一个人瞄准(有可能瞄准自己).然后他们按某个顺序开枪,且任意时刻只有一个人开 ...

  5. [POI2008]枪战Maf题解

    问题 C: [POI2008]枪战Maf 时间限制: 1 Sec  内存限制: 256 MB 题目描述 有n个人,每个人手里有一把手枪.一开始所有人都选定一个人瞄准(有可能瞄准自己).然后他们按某个顺 ...

  6. BZOJ1124 [POI2008]枪战Maf[贪心(证明未完成)+拓扑排序]

    吐槽:扣了几个小时,大致思路是有了,但是贪心的证明就是不会, 死磕了很长时间,不想想了,结果码代码又不会码.. 深深体会到自己码力很差,写很多行还没写对,最后别人代码全一二十行,要哭了 以下可能是个人 ...

  7. 枪战Maf[POI2008]

    题目描述 有n个人,每个人手里有一把手枪.一开始所有人都选定一个人瞄准(有可能瞄准自己).然后他们按某个顺序开枪,且任意时刻只有一个人开枪.因此,对于不同的开枪顺序,最后死的人也不同. 输入 输入n人 ...

  8. bzoj1124[POI2008]枪战maf

    这代码快写死我了.....死人最多随便推推结论.死人最少,每个环可以单独考虑,每个环上挂着的每棵树也可以分别考虑.tarjan找出所有环,对环上每个点,求出选它和不选它时以它为根的树的最大独立集(就是 ...

  9. 【BZOJ】1124: [POI2008]枪战Maf

    题意 \(n(n < 1000000)\)个人,每个人\(i\)指向一个人\(p_i\),如果轮到\(i\)了且他没死,则他会将\(p_i\)打死.求一种顺序,问死的人最少和最多的数目. 分析 ...

  10. BZOJ1124 POI2008枪战Maf(环套树+贪心)

    每个点出度都为1,可以发现这张图其实是个环套树森林,树中儿子指向父亲,环上边同向. 首先自环肯定是没救的,先抬出去. 要使死亡人数最多的话,显然若一个点入度为0其不会死亡,而一个孤立的环至少会留下一个 ...

随机推荐

  1. 某宝购入牙膏厂U后其售后事宜的思虑

    近日某宝购入i7 9700k,    突然想去保修的问题(没有发票,要发票比京东自营贵200左右, 不要则便宜100左右),由于确实囊中幸亏所以还是选择了某宝,东西到手后突然想起这东西没票是否可以保修 ...

  2. 《Python数据可视化之matplotlib实践》 源码 第一篇 入门 第四章

    图 4.1 import matplotlib import matplotlib.pyplot as plt import numpy as np # 设置matplotlib正常显示中文和负号 m ...

  3. Python网络连接request报错:OSError: [Errno 113] No route to host

    报错: (pytorch) devil@Monster:~$ huggingface-cli login _| _| _| _| _|_|_| _|_|_| _|_|_| _| _| _|_|_| _ ...

  4. PHP 字符串大小写操作

    PHP为我们提供了字符串中大小写字母转换的函数, strtoupper()将指定的字符全部转换为大写: strtolower()将北定的字符都转换成小写: ucwords()将指定字符串中每个单词的首 ...

  5. 清除 Nuxt 数据缓存:clearNuxtData

    title: 清除 Nuxt 数据缓存:clearNuxtData date: 2024/8/6 updated: 2024/8/6 author: cmdragon excerpt: 摘要:本文详细 ...

  6. 2024-08-17:用go语言,给定一个从0开始的整数数组nums和一个整数k, 每次操作可以删除数组中的最小元素。 你的目标是通过这些操作,使得数组中的所有元素都大于或等于k。 请计算出实现这个目

    2024-08-17:用go语言,给定一个从0开始的整数数组nums和一个整数k, 每次操作可以删除数组中的最小元素. 你的目标是通过这些操作,使得数组中的所有元素都大于或等于k. 请计算出实现这个目 ...

  7. .net core下使用事件总线

            随着微服务的火热,DDD(领域驱动设计模式)思想风起云涌,冲击着整个软件生态系统.其中,事件总线那是必须知道的了,于是我便抱着一个学习DDD的心态搭建了一个博客网站,目前该网站正在建设 ...

  8. 基于surging 如何利用peerjs进行语音视频通话

    一 . 概述 PeerJS 是一个基于浏览器WebRTC功能实现的js功能包,简化了WebrRTC的开发过程,对底层的细节做了封装,直接调用API即可,再配合surging 协议组件化从而做到稳定,高 ...

  9. pc 移动端 双端切换-路由判断

    该封装主要以分类形式,实现对路由的简易区分.便于项目管理. 创建好项目,勾选路由插件,会自动生成 router文件夹与index.ts . index.ts 初始内容 创建项目 自动生成的router ...

  10. 部署在阿里云上的项目收到了阿里云发送的shiro漏洞

    编辑 ​ 还记得在十月份凯哥发布过一篇修改若依系统编辑器的文章,然后为了方便大家浏览,凯哥就部署在服务器上了,结果,没想到最近收到了阿里云漏洞扫描通知: 编辑 ​ 如果不修改的话:对于长期存在安全隐患 ...