题目描述

【背景】
ZCC又在打Isaac。这次他打通了宝箱关进入了表箱关。
【题目描述】
表箱关有一个房间非常可怕,它由n个变异天启组成。
每个天启都会在进入房间后吐出绿弹并炸向某一个位置且范围内只有一个天启。若该位置的天启已经死亡则没有事情发生,否则该位置的天启会死亡。每个天启只能且必须吐一次绿弹(除非在它吐弹以前他就挂了)。
绿弹的飞行速度很快,在某个绿弹落地之前不会有新的绿弹被吐出。
虽然房间的天启位置和吐弹位置固定,但是吐弹顺序是随机的,所以ZCC不能很好地制定策略。
现在ZCC想知道,最少和最多有几个天启被干掉。

输入

第一行n表示天启个数。
第二行n个数ai表示i号天启的目标是ai。
注意:行末有一个空格。

输出

一行两个数表示最少和最多有几个天启被干掉。

样例输入

8
2 3 2 2 6 7 8 5

样例输出

3 5

提示

 
 
 
 
 
 
 
 

【样例解释】

YYHS-NOIP2017Training0928-ZCC loves Isaac-LMLPHP

【数据范围】

测试点编号

n的范围

其他

测试点编号

n的范围

其他

1

=10

随机生成

6

=10000

处于一个联通块

2

=100

随机生成

7

=100000

所有天启都能被炸到

3

=1000

随机生成

8

=100000

4

=1000

随机生成

9

=1000000

处于一个联通块

5

=10000

随机生成

10

=1000000

题解

这道题刚开始弄了我很久,后来发现每个点只会向后连一条边,这样就少了很多种特殊情况

这道题要我们求最少和最多被干掉的天启数量

我们把图的情况想象成  单独一个环和其他的情况

求最少的数量

我们对于单独的环最少一定是干掉  (环的长度+1)/2  个天启

而对于其他的情况 我们就从入度为0的点开始往后找,把这些点标记掉,再找入度为0的点,这样继续,在判断的时候把干掉的加起来即可

求最大的数量

单独的环最后一定是只剩下一个(自环除外),自环就要特判一下

其他情况就只有入度为0的点存活,其他都会被干掉

怎么求单独的环呢?这个问题确实弄了我很久

其实我们可以先把入度为0的点向后找,边找边标记,一直找到找不了为止,这样剩下来的点就是构成环的点了

 #include<bits/stdc++.h>
#define N 1000005
using namespace std;
int n,cnt,Max,Min,k,u,v;
int a[N],in[N],s[N],flag[N];
int calc(int u){
flag[u]=;
int sum=,v=u;
while (!flag[a[v]]){
flag[a[v]]=;
sum++;
v=a[v];
}
if (sum==) sum=;
return sum;
}
int main(){
scanf("%d",&n);
for (int i=;i<=n;i++){
scanf("%d",&a[i]);
in[a[i]]++;
}
for (int i=;i<=n;i++)
if (!in[i]) s[++cnt]=i,flag[i]=true;
int num=;
while (num<=cnt){
k=s[num];
u=a[k];
if (!flag[u]){
Min++; flag[u]=true;
v=a[u];
in[v]--;
if (!in[v]){
s[++cnt]=v;
flag[v]=true;
}
}
num++;
}
for (int i=;i<=n;i++)
if (!flag[i]) Min+=(calc(i)+)/;
memset(flag,,sizeof(flag));
for (int i=;i<=n;i++)
in[a[i]]++;
cnt=;
for (int i=;i<=n;i++)
if (!in[i]) s[++cnt]=i,flag[i]=true;
num=;
while (num<=cnt){
k=s[num];
u=a[k];
if (!flag[u]){
Max++; flag[u]=true;
s[++cnt]=u;
}
num++;
}
for (int i=;i<=n;i++)
if (!flag[i]) Max+=calc(i)-;
printf("%d %d\n",Min,Max);
return ;
}
05-08 08:25