【每日一题】洛谷P4819 [中山市选]杀人游戏

[中山市选]杀人游戏

题目描述

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

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

输入格式

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

注:原文zz敏感内容已替换

输出格式

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

样例 #1

样例输入 #1

5 4 
1 2 
1 3 
1 4 
1 5

样例输出 #1

0.800000

提示

题解

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

对于 100 % 100\% 100%的数据有 1 ≤ N ≤ 100000 , 0 ≤ M ≤ 300000 1≤N≤100000,0≤M≤300000 1N100000,0M300000

题意简述

给出一个有向图,表示每个人的认识关系,所有人中有一个杀手,询问正常人时他会说出自己认识的人的身份,询问杀手则会被杀死,问活下来并知道杀手的最大概率是多少。

题目分析

在此题中,由于每个人是杀手的概率均等,假设抽查了 p p p 个人,活下来的概率则为 1 − p n 1 - \frac{p}{n} 1np,若要让该概率最大,我们可以转化为求最小的 p p p

由于每个人的认识关系为一张无向图,因此只要对一个人发出询问,就能知道他所连边的人的身份,进而推出他能到达的人的身份,因此我们可以想到一个贪心策略:不选有入度点,否则选取能到达他的点一定更优。

但图中可能有环的存在,加上图不连通,就可能出现必须选一个有入度点的情况,于是我们考虑对图进行操作,注意到如果两点在一个强连通分量中,那么选取这两个点是等效的,因此我们可以将图缩点,进而统计无入度点的个数,因为此处将环也缩成了一个节点,因此不会有上述情况且更方便操作。

但我们还需要注意一种特殊情况,当一个强连通分量的大小为 1 且能直接到达的点都起码有两个入度时,这个点不用选,因为其他点检查过后只剩他一个没有检查,可以直接推测出这个点的身份,但是因为我们使用了推测法,这样的点只能存在一个。

总上,我们建图后求 scc,统计无入度点和特殊情况即可。

代码

#include<iostream>
#include<cstdio>
#include<algorithm>
#include<cstring>
#include<vector>
#include<map>
using namespace std;
map<pair<int,int> ,int > t;
struct Node{
	int u,v;
}edg[300001];
vector<int> l[300001];
vector<int> tr[300001];
int dfn[300001],low[300001];
int c[300001],q[300001],ins[300001];
int in[300001],ou[300001];
int siz[300001];
int tail,tot,cnt;
void tarjan(int x){
	dfn[x]=low[x]=++tot;
	q[++tail]=x;
	ins[x]=1;
	for(int i=0;i<l[x].size();i++){
		int v=l[x][i];
		if(!dfn[v]){
			tarjan(v);
			low[x]=min(low[x],low[v]);
		}
		else if(ins[v])low[x]=min(low[x],dfn[v]);
	}
	if(low[x]==dfn[x]){
		int y;
		cnt++;
		do{
			y=q[tail--];
			c[y]=cnt;
			ins[y]--;
			siz[cnt]++;
		}while(y!=x);
	}
}
int main(){
	int n,m,x,y;
	cin>>n>>m;
	for(int i=1;i<=m;i++){
		cin>>x>>y;
		l[x].push_back(y);
		edg[i].u=x;edg[i].v=y;
	}
	for(int i=1;i<=n;i++) if(!dfn[i])tarjan(i);
	for(int i=1;i<=m;i++){
		if(c[edg[i].u]==c[edg[i].v])continue;
		pair<int,int> tmp;
		tmp.first=c[edg[i].u];tmp.second=c[edg[i].v];
		if(t[tmp])continue;
		else t[tmp]=1;
		in[c[edg[i].v]]++;
		ou[c[edg[i].u]]++;
		tr[c[edg[i].u]].push_back(c[edg[i].v]);
	}
	int ans=0;
	for(int i=1;i<=cnt;i++) if(in[i]==0) ans++;
	for(int i=1;i<=cnt;i++){
		if(in[i] || siz[i]>1) continue;
		int flag=0;
		for(int j=0;j<tr[i].size();j++){
			if(in[tr[i][j]]<2){
				flag=1;
				break;
			}
		}
		if (!flag){
			ans--;
			break;
		}
	}
	double p=(double)(n-ans)*1.0/n;
	printf("%.6lf",p);
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
洛谷P2791是一道关于幼儿园篮球的题目。题目描述如下: 在幼儿园里,小朋友们正在进行篮球比赛。每个小朋友都有一个篮球,他们按照顺序依次投篮。每个小朋友投篮时,可以择将篮球投给左边的小朋友或者右边的小朋友。每个小朋友投篮得分的规则如下: 1. 如果一个小朋友左右两边的小朋友都没有投篮过,那么他的得分为1; 2. 如果一个小朋友左边的小朋友投篮过,但右边的小朋友没有投篮过,那么他的得分为左边小朋友的得分加1; 3. 如果一个小朋友右边的小朋友投篮过,但左边的小朋友没有投篮过,那么他的得分为右边小朋友的得分加1; 4. 如果一个小朋友左右两边的小朋友都投篮过,那么他的得分为左边小朋友和右边小朋友得分的最大值加1。 现在给定每个小朋友投篮的顺序,请你计算每个小朋友的得分。 例如,给定投篮顺序为[1, 0, 1, 0, 1],则第一个小朋友的得分为1,第二个小朋友的得分为2,第三个小朋友的得分为1,第四个小朋友的得分为2,第五个小朋友的得分为1。 你可以通过编写程序来解决这个问题。具体的解题思路可以参考以下步骤: 1. 创建一个数组scores,用来存储每个小朋友的得分; 2. 遍历投篮顺序数组,对于每个小朋友,根据上述规则计算他的得分,并将得分存入scores数组中; 3. 最后输出scores数组即可。 希望以上解答对你有帮助!如果你还有其他问题,请继续提问。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值