AcWing 258. 石头剪子布(扩展域并查集)

2024-04-15 23:08

本文主要是介绍AcWing 258. 石头剪子布(扩展域并查集),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

N个小朋友(编号为0,1,2,…,N-1)一起玩石头剪子布游戏。

其中一人为裁判,其余的人被分为三个组(有可能有一些组是空的),第一个组的小朋友只能出石头,第二个组的小朋友只能出剪子,第三个组的小朋友只能出布,而裁判可以使用任意手势。

你不知道谁是裁判,也不知道小朋友们是怎么分组的。

然后,孩子们开始玩游戏,游戏一共进行M轮,每轮从N个小朋友中选出两个小朋友进行猜拳。

你将被告知两个小朋友猜拳的胜负结果,但是你不会被告知两个小朋友具体使用了哪种手势。

比赛结束后,你能根据这些结果推断出裁判是谁吗?

如果可以的话,你最早在第几轮可以找到裁判。

输入格式
输入可能包含多组测试用例

每组测试用例第一行包含两个整数N和M。

接下来M行,每行包含两个整数a,b,中间夹着一个符号(‘>’,’=’,’<’),表示一轮猜拳的结果。

两个整数为小朋友的编号,”a>b”表示a赢了b,”a=b”表示a和b平手,”a<b”表示a输给了b。

输出格式
每组测试用例输出一行结果,如果找到裁判,且只能有一个人是裁判,则输出裁判编号和确定轮数。

如果找到裁判,但裁判的人选多于1个,则输出“Can not determine”。

如果根据输入推断的结果是必须没有裁判或者必须有多个裁判,则输出“Impossible”。

具体格式可参考样例。

数据范围
1≤N≤500,
0≤M≤2000
输入样例:
3 3
0<1
1<2
2<0
3 5
0<1
0>1
1<2
1>2
0<2
4 4
0<1
0>1
2<3
2>3
1 0
输出样例:
Can not determine
Player 1 can be determined to be the judge after 4 lines
Impossible
Player 0 can be determined to be the judge after 0 lines

思路:
每个点拆成三份
x x x:同类域
x + n x+n x+n:能打败的域
x + 2 ∗ n x+2*n x+2n:被打败的域

按照这个思路来处理点之间的关系。
我们枚举裁判点,然后出现裁判的不等式就去掉,如果中间没有矛盾就说明这是一个合法裁判。统计合法裁判的数量和出现的位置。

#include <cstdio>
#include <cstring>
#include <algorithm>
#include <vector>
#include <map>
using namespace std;
typedef long long ll;
const int INF = 0x3f3f3f3f;
const int maxn = 2005;
struct Expr{int x,y;char op;
}expr[maxn];
int n,m;
int fa[maxn],ind[maxn];
int findset(int x) {if(fa[x] == x) return x;return fa[x] = findset(fa[x]);
}
void Union(int x,int y) {int rx = findset(x),ry = findset(y);if(rx != ry) {fa[rx] = ry;}
}
//x:同类,x+n:能打败,x+2*n:被打败
bool conflict(Expr&now) { //是否发生冲突int x = now.x,y = now.y;if(now.op == '=') {if(findset(x) == findset(y + n) || findset(x + n) == findset(y)) return true;Union(x,y);Union(x + n,y + n);Union(x + 2 * n,y + 2 * n);} else if(now.op == '>') {if(findset(x) == findset(y) || findset(x) == findset(y + n)) return true;Union(x,y + 2 * n);Union(x + n,y);Union(x + 2 * n,y + n);} else if(now.op == '<') {if(findset(x) == findset(y) || findset(x) == findset(y + 2 * n)) return true;Union(x,y + n);Union(x + n,y + 2 * n);Union(x + 2 * n,y);}return false;
}int main() {while(~scanf("%d%d",&n,&m)) {for(int i = 0;i < m;i++) {scanf("%d%c%d",&expr[i].x,&expr[i].op,&expr[i].y);}for(int i = 0;i < n;i++) ind[i] = 0;int cnt = 0;//裁判个数int id = 0;//裁判位置for(int i = 0;i < n;i++) { //枚举裁判for(int j = 0;j < n * 3;j++) fa[j] = j;int flag = 1;for(int j = 0;j < m;j++) {if(expr[j].x == i || expr[j].y == i) continue;if(conflict(expr[j])) {ind[i] = j + 1;flag = 0;break;}}if(flag) {cnt++;id = i;}}
//        printf("DEBUG %d\n",cnt);if(cnt == 0) {printf("Impossible\n");} else if(cnt == 1) {int pos = 0;for(int i = 0;i < n;i++) pos = max(pos,ind[i]);printf("Player %d can be determined to be the judge after %d lines\n",id,pos);} else {printf("Can not determine\n");}}return 0;
}

这篇关于AcWing 258. 石头剪子布(扩展域并查集)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/907184

相关文章

csu 1446 Problem J Modified LCS (扩展欧几里得算法的简单应用)

这是一道扩展欧几里得算法的简单应用题,这题是在湖南多校训练赛中队友ac的一道题,在比赛之后请教了队友,然后自己把它a掉 这也是自己独自做扩展欧几里得算法的题目 题意:把题意转变下就变成了:求d1*x - d2*y = f2 - f1的解,很明显用exgcd来解 下面介绍一下exgcd的一些知识点:求ax + by = c的解 一、首先求ax + by = gcd(a,b)的解 这个

科研绘图系列:R语言扩展物种堆积图(Extended Stacked Barplot)

介绍 R语言的扩展物种堆积图是一种数据可视化工具,它不仅展示了物种的堆积结果,还整合了不同样本分组之间的差异性分析结果。这种图形表示方法能够直观地比较不同物种在各个分组中的显著性差异,为研究者提供了一种有效的数据解读方式。 加载R包 knitr::opts_chunk$set(warning = F, message = F)library(tidyverse)library(phyl

poj 1182 并查集 食物链类

题意: 有n只动物,分别编号1....n。所有动物都属于A,B,C中的一种,已知A吃B,B吃C,C吃A。 按顺序给出下面两种共K条信息: 1. x 和 y 属于同一类。 2. x 吃 y 。 然而这些信息可能会出错,有可能有的信息和之前给出的信息矛盾,也有的信息可能给出的 x 和 y 不在n的范围内。 求k条信息中有多少条是不正确的。 解析: 对于每只动物,创建3个元素 i

Spring框架5 - 容器的扩展功能 (ApplicationContext)

private static ApplicationContext applicationContext;static {applicationContext = new ClassPathXmlApplicationContext("bean.xml");} BeanFactory的功能扩展类ApplicationContext进行深度的分析。ApplicationConext与 BeanF

POJ1988带权并查集

M u v 将u所在的堆移动到v所在的堆的上面 C u 求u的下面有多少块 带权并查集 import java.io.BufferedReader;import java.io.InputStream;import java.io.InputStreamReader;import java.io.PrintWriter;import java.math.BigInteger;i

POJ1703带权并查集

D: u v  u与v在不同的集合 A: u v  查询u与v的关系 1)压缩路径过程        fu->root   0  1  u-fu 0                 0  1   1                 1  0 2)合并过程 fu->fv      u->fu   0 1   v->fv 0            1 0 1            0

PHP7扩展开发之数组处理

前言 这次,我们将演示如何在PHP扩展中如何对数组进行处理。要实现的PHP代码如下: <?phpfunction array_concat ($arr, $prefix) {foreach($arr as $key => $val) {if (isset($prefix[$key]) && is_string($val) && is_string($prefix[$key])) {$arr[

PHP7扩展开发之字符串处理

前言 这次,我们来看看字符串在PHP扩展里面如何处理。 示例代码如下: <?phpfunction str_concat($prefix, $string) {$len = strlen($prefix);$substr = substr($string, 0, $len);if ($substr != $prefix) {return $prefix." ".$string;} else

PHP7扩展开发之类型处理

前言 这次,我们将演示如何在PHP扩展中如何对类型进行一些操作。如,判断变量类型。要实现的PHP代码如下: <?phpfunction get_size ($value) {if (is_string($value)) {return "string size is ". strlen($value);} else if (is_array($value)) {return "array si

PHP7扩展开发之依赖其他扩展

前言 有的时候,我们的扩展要依赖其他扩展。比如,我们PHP的mysqli扩展就依赖mysqlnd扩展。这中情况下,我们怎么使用其他扩展呢?这个就是本文讲述的内容。 我们新建立一个扩展,名字叫 demo_dep , 依赖之前的say扩展。 在demo_dep扩展中,我们实现demo_say方法。这个方法调用say扩展的say方法。 代码 基础代码 确保say扩展的头文件正确安装到了php