[中山市选]杀人游戏 Tarjan+概率

[中山市选]杀人游戏 Tarjan+概率

题目描述

? 一位冷血的杀手潜入\(Na\)-\(wiat\),并假装成平民。警察希望能在\(N\)个人里面,查出谁是杀手。警察能够对每一个人进行查证,假如查证的对象是平民,他会告诉警察,他认识的人,谁是杀手,谁是平民。假如查证的对象是杀手,杀手将会把警察干掉。现在警察掌握了每一个人认识谁。每一个人都有可能是杀手,可看作他们是杀手的概率是相同的。

问:根据最优的情况,保证警察自身安全并知道谁是杀手的概率最大是多少?

输入输出格式

输入格式:

? 第一行有两个整数 \(N,M\)。 接下来有 \(M\) 行,每行两个整数 \(x,y\),表示 \(x\) 认识 \(y\)(\(y\) 不一定认识 \(x\) ,例如\(President\)同志) 。

输出格式:

? 仅包含一行一个实数,保留小数点后面 \(6\) 位,表示最大概率。

输入输出样例

输入样例#1:

5 4
1 2
1 3
1 4
1 5 

输出样例#1:

0.800000

说明

? 警察只需要查证\(1\)。假如\(1\)是杀手,警察就会被杀。假如\(1\)不是杀手,他会告诉警察\(2,3,4,5\)谁是杀手。而\(1\)是杀手的概率是\(0.2\),所以能知道谁是杀手但没被杀的概率是\(0.8\)。

题解

首先想到的是\(Tarjan\)缩点,因为缩点之后我们只需要查询环中的一个人就可以知道整个环的信息。

而且若一个环又指向另一个环,那么我们要是选择一个缩点后入度为0的点,即可以把其连通的若干个环的信息全部了解。

但是这样还有还有一些错误:

举个例子,若一共有\(N\)个人,我们查询了一些人,使得我们知道了\(N\)-\(1\)个人的身份(都是平民,否则你就挂了),那么剩下的一个人一定是杀手(排除法),那么我们就可以少去查询一个人。

所以,我们要去寻找一个大小为1且入度为0的点((缩点后)就是一个与世隔绝的人),并且他指向的点入度不为1(就是能只通过这个人来获取信息),那么我们就可以少一次询问,则标记\(flag=1\)。

综上,若\(flag==1,ans=1\)-\((缩点后入度为0的点\)-\(1)/n\).

若\(flag==0,ans=1\)-\((缩点后入度为0的点)/n\).

下面是代码欧。。。

#include<iostream>
#include<algorithm>
#include<cstdio>
#include<cstring>
#include<cctype>
#define ll long long
#define R register
#define M 300005
#define N 100005
using namespace std;
template<typename T>inline void read(T &a){
    char c=getchar();T x=0,f=1;
    while(!isdigit(c)){if(c=='-')f=-1;c=getchar();}
    while(isdigit(c)){x=(x<<1)+(x<<3)+c-'0';c=getchar();}
    a=f*x;
}
int n,m,h[N],vis[N],low[N],dfn[N],num,tot,sta[N],top,cnt,col[N],siz[N];
int u[M],v[M],in[N],flag,ans;
struct node{
    int nex,to;
}edge[M];
inline void add(R int u,R int v){
    edge[++tot].nex=h[u];
    edge[tot].to=v;
    h[u]=tot;
}
inline void tarjan(R int x){
    dfn[x]=low[x]=++num;
    sta[++top]=x;vis[x]=1;
    for(R int i=h[x];i;i=edge[i].nex){
        R int xx=edge[i].to;
        if(!dfn[xx]){
            tarjan(xx);
            low[x]=min(low[x],low[xx]);
        }
        else if(vis[xx])low[x]=min(low[x],dfn[xx]);
    }
    if(low[x]==dfn[x]){
        R int now=-1;cnt++;
        while(now!=x){
            now=sta[top];
            top--;
            vis[now]=0;
            siz[cnt]++;
            col[now]=cnt;
        }
    }
}
int main(){
    read(n);read(m);
    for(R int i=1;i<=m;i++)
        read(u[i]),read(v[i]),add(u[i],v[i]);
    for(R int i=1;i<=n;i++)
        if(!dfn[i])tarjan(i);
    memset(h,0,sizeof(h));tot=0;
    for(R int i=1;i<=m;i++)
        if(col[u[i]]!=col[v[i]]){
            in[col[v[i]]]++;
            add(col[u[i]],col[v[i]]);//缩点后重新建图
        }
    for(R int i=1;i<=cnt;i++){
        if(!flag&&!in[i]&&siz[i]==1){//找上述符合条件的点
            R int pd=0;
            for(R int j=h[i];j;j=edge[j].nex){
                R int xx=edge[j].to;
                if(in[xx]==1)pd=1;
            }
            if(!pd)flag=1;
        }
        if(!in[i])ans++;
    }
    if(flag)ans--;//若标记则可以少一次询问
    printf("%.6f\n",1.0-(double)ans/(double)n);
    return 0;
}

原文地址:https://www.cnblogs.com/ZAGER/p/9813448.html

时间: 2024-10-16 07:12:00

[中山市选]杀人游戏 Tarjan+概率的相关文章

P4819 [中山市选]杀人游戏

题目描述 一位冷血的杀手潜入Na-wiat,并假装成平民.警察希望能在NN个人里面,查出谁是杀手.警察能够对每一个人进行查证,假如查证的对象是平民,他会告诉警察,他认识的人,谁是杀手,谁是平民.假如查证的对象是杀手,杀手将会把警察干掉.现在警察掌握了每一个人认识谁.每一个人都有可能是杀手,可看作他们是杀手的概率是相同的. 问:根据最优的情况,保证警察自身安全并知道谁是杀手的概率最大是多少? Solution 首先缩点, 然后需要把缩完点后的DAG上每个入度为0的点都询问一次才行. 但是有一种特殊

【BZOJ2438】[中山市选2011]杀人游戏 Tarjan

[BZOJ2438][中山市选2011]杀人游戏 Description 一位冷血的杀手潜入 Na-wiat,并假装成平民.警察希望能在 N 个人里面,查出谁是杀手. 警察能够对每一个人进行查证,假如查证的对象是平民,他会告诉警察,他认识的人, 谁是杀手, 谁是平民. 假如查证的对象是杀手, 杀手将会把警察干掉. 现在警察掌握了每一个人认识谁. 每一个人都有可能是杀手,可看作他们是杀手的概率是相同的. 问:根据最优的情况,保证警察自身安全并知道谁是杀手的概率最大是多少? Input 第一行有两个

【BZOJ-2438】杀人游戏 Tarjan + 缩点 + 概率

2438: [中山市选2011]杀人游戏 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 1638  Solved: 433[Submit][Status][Discuss] Description 一位冷血的杀手潜入 Na-wiat,并假装成平民.警察希望能在 N 个人里面,查出谁是杀手. 警察能够对每一个人进行查证,假如查证的对象是平民,他会告诉警察,他认识的人, 谁是杀手, 谁是平民. 假如查证的对象是杀手, 杀手将会把警察干掉. 现在警察掌

bzoj2438 杀人游戏 Tarjan强联通

[bzoj2438][中山市选2011]杀人游戏 Description 一位冷血的杀手潜入 Na-wiat,并假装成平民.警察希望能在 N 个人里面,查出谁是杀手.警察能够对每一个人进行查证,假如查证的对象是平民,他会告诉警察,他认识的人, 谁是杀手, 谁是平民. 假如查证的对象是杀手, 杀手将会把警察干掉.现在警察掌握了每一个人认识谁.每一个人都有可能是杀手,可看作他们是杀手的概率是相同的.问:根据最优的情况,保证警察自身安全并知道谁是杀手的概率最大是多少? Input 第一行有两个整数 N

BZOJ 2438 中山市选 2011 杀人游戏 Tarjan

题目大意:给出一张有向人物关系图,告诉你谁认识谁,认识具有传递性.其中有一个人是犯人.现在警察要调查谁是犯人.他可以问任何人.但是如果他问到了犯人,那么它就会死.如果他问到的一个人认识犯人,这个人就会告诉警察谁是犯人.问警察保证自身安全并知道犯人是谁的概率最大是多少. 思路:这个题前一阵子重测了,加强了数据,卡掉了网上一片AC代码.. 正解并不是很难想.首先先缩点,整个图变成拓扑图,之后会出现一些类似根的东西,这些scc入度为0,只要警察询问了这些scc每一个中的任意一个,就肯定能知道谁是犯人.

BZOJ 2438 中山市选2011 杀人游戏 Tarjan

题目大意:有n个人,其中一个是杀手,可以询问一些人,如果是杀手就会死,如果是平民,他会告诉你他认识的人中有谁是杀手有谁是平民 警告:数据有误,请谨慎提交! 易知如果我需要访问x个人,那么答案就是1-x/n 我们需要访问最少的人 如果我访问的人是平民,那么这个点所有的后继我都能知道 于是Tarjan缩点之后入度为零的点就是答案 但是还有一个问题 比如说这组样例 3 1 1 2 我访问了1,那么1和2是不是凶手我就都知道了 既然只有三个人,我知道1和2是不是凶手,那么3也一定知道 没必要去访问3 于

[中山市选2011]杀人游戏

[中山市选2011]杀人游戏 时间限制: 1 Sec  内存限制: 128 MB提交: 64  解决: 33[提交][状态][讨论版] 题目描述 一位冷血的杀手潜入 Na-wiat,并假装成平民.警察希望能在 N 个人里面,查出谁是杀手. 警察能够对每一个人进行查证,假如查证的对象是平民,他会告诉警察,他认识的人, 谁是杀手, 谁是平民. 假如查证的对象是杀手, 杀手将会把警察干掉. 现在警察掌握了每一个人认识谁. 每一个人都有可能是杀手,可看作他们是杀手的概率是相同的. 问:根据最优的情况,保

bzoj2438: [中山市选2011]杀人游戏(强联通+特判)

2438: [中山市选2011]杀人游戏 题目:传送门 简要题意: 给出n个点,m条有向边,进行最少的访问并且可以便利(n-1)个点,求这个方案成功的概率 题解: 一道非常好的题目! 题目要知道最大的存活概率,那么也就是找到直接找到杀手的最小概率 那么我们采用强联通缩点: 统计每个联通分量的入度,如果入度为0(证明除此联通分量里的点,没有人可以知道连通分量里的信息,那就一定要先选一个人访问),那么sum++(因为依据题意,假如问到连通分量里的任意一个人,只要ta不是杀手,那么一定可以安全的遍历强

BZOJ2464: 中山市选[2009]小明的游戏

2464: 中山市选[2009]小明的游戏 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 280  Solved: 124[Submit][Status] Description 小 明最近喜欢玩一个游戏.给定一个n * m的棋盘,上面有两种格子#和@.游戏的规则很简单:给定一个起始位置和一个目标位置,小明每一步能向上,下,左,右四个方向移动一格.如果移动到同一类 型的格子,则费用是0,否则费用是1.请编程计算从起始位置移动到目标位置的最小花费.