[NOIP2003 提高组] 侦探推理(C++,字符串)

news2024/11/13 10:13:40

题目描述

明明同学最近迷上了侦探漫画《柯南》并沉醉于推理游戏之中,于是他召集了一群同学玩推理游戏。游戏的内容是这样的,明明的同学们先商量好由其中的一个人充当罪犯(在明明不知情的情况下),明明的任务就是找出这个罪犯。接着,明明逐个询问每一个同学,被询问者可能会说:

证词中出现的其他话,都不列入逻辑推理的内容。

明明所知道的是,他的同学中有 N N N 个人始终说假话,其余的人始终说真。

现在,明明需要你帮助他从他同学的话中推断出谁是真正的凶手,请记住,凶手只有一个!

输入格式

输入由若干行组成。

第一行有三个整数, M , N M,N M,N P P P M M M 是参加游戏的明明的同学数, N N N 是其中始终说谎的人数, P P P 是证言的总数。

接下来 M M M 行,每行是明明的一个同学的名字(英文字母组成,没有空格,全部大写)。

往后有 P P P 行,每行开始是某个同学的名宇,紧跟着一个冒号和一个空格,后面是一句证词,符合前表中所列格式。证词每行不会超过 250 250 250 个字符。

输入中不会出现连续的两个空格,而且每行开头和结尾也没有空格。

输出格式

如果你的程序能确定谁是罪犯,则输出他的名字;如果程序判断出不止一个人可能是罪犯,则输出 Cannot Determine;如果程序判断出没有人可能成为罪犯,则输出 Impossible

样例 #1

样例输入 #1

3 1 5
MIKE
CHARLES
KATE
MIKE: I am guilty.
MIKE: Today is Sunday.
CHARLES: MIKE is guilty.
KATE: I am guilty.
KATE: How are you??

样例输出 #1

MIKE

提示

对于 100 % 100\% 100% 数据,满足 1 ≤ M ≤ 20 1\le M\le 20 1M20 0 ≤ N ≤ M 0\le N\le M 0NM 1 ≤ P ≤ 100 1\le P\le 100 1P100

【题目来源】

NOIP 2003 提高组第二题

解题思路:

这道题其实并不难,考察的是最基本的模拟和枚举

模拟侦探推理的过程,关键在于找出矛盾的证词

枚举每一个可能的罪犯和一周中的七天,然后判断是否符合条件

下面的代码虽然很繁琐但是思路是很清晰的,大致思路如下:

枚举每一个人、每一天,然后开始推理

每轮开始假设每个人都是未知状态-1,说真话的人是1,说假话的人是0

(1)对于每一句话进行判断,根据现有条件判断这句话是真话还是假话

(2)判断说话人的状态

(3)如果状态是未知状态,则赋予其状态,并累加说真话(假话)的人数

(4)如果状态是已知的,那么判断其说的话是否与状态相矛盾(如说假话的人说了真话),如果矛盾,那么本轮推理结束,把end_index标志赋值为true

结束了?当然没有

思路很好想,但本题易错点并不在这里,而是在于从字符串中提取信息,要不然这篇题解也不会以字符串为标题了

首先最重要的一点是绝对不要假设字符串会以某种顺序出现

比如说你是这么处理字符串的:把每一个单词输入后装到容器里面,然后如果容器中的第一个单词是I,第三个单词不是guilty,那么你认为这句话是I am not guilty.

但这就错了

输入的数据甚至可能不是单词,如I am not guity.

然后,你不能假设字符串中出现了Name is guilty.就判断这句话说的是Name是罪犯

因为玩家们的名字可能是这样的AAAAAAAAAA

那么A is guilty.AA is guilty.中均含有A is guilty.

你就又输出错误答案了

当然,还有就是你不能认为他们说的话不是废话

比如,一个人可以说How are you?What's your name?

甚至一个人可能不会说话

所以一个人的状态在每轮推理结束后仍然可能是-1,也就是未定状态,你可以把他归于任何一类之中

所以判断一轮推断是否有效是根据说真话的人 <= m - n && 说假话的人 <= n

最后再说一点就是换行符的问题

Linux中的换行符和Windows的换行符是不同的,要注意判题系统是什么操作系统

Linux下的换行符是\r,而Windows下的换行符是\r + \n

这会导致什么问题呢?

要知道,对于Linux的getchar()来说,Windows的换行符是两个字符,也就会导致靠getchar()判断循环终止条件的代码的崩溃

以上就是错误点啦~(都是我错的QAQ)

p.s.:关于代码中的sentence结构体中的info数组

对于人:info[0]是人的索引,info[1]是这句话对此人是否为罪犯的判断

对于日期:info[0]0(固定),info[1]是这句话对日期的判断

最后,AC代码入戏

#include <iostream>
#include <map>
#include <vector>
#include <string>
#include <string.h>
using namespace std;
const int max_m = 20;

int m, n, p;
map<string, int>name2index;//name to index
struct sent { int info[2]; };
vector<sent>sents[max_m + 1];//index to sentences
map<string, int>week2index;//week to index
int states[max_m + 1];


void get_sent(int index) {
	string s;
	sent sentence;

	getline(cin, s);//读入句子
	
	for (map<string, int>::iterator it = week2index.begin(); it != week2index.end(); it++) {//日期类型
		if (s.find(it->first) != -1) {
			sentence.info[0] = 0;
			sentence.info[1] = it->second;
			sents[index].push_back(sentence);
			return;
		}
	}
	
	if (s.find("I am guilty.") != -1) {//罪人指定一
		sentence.info[0] = index;
		sentence.info[1] = 1;
		sents[index].push_back(sentence);
		return;
	}
	else if (s.find("I am not guilty.") != -1) {
		sentence.info[0] = index;
		sentence.info[1] = 0;
		sents[index].push_back(sentence);
		return;
	}

	int ret, ret2;
	if ((ret = s.find(" is guilty.")) != -1) {//罪人指定二
		ret2 = s.rfind(' ', ret - 1);
		sentence.info[0] = name2index[s.substr(ret2 + 1, ret - 1)];
		sentence.info[1] = 1;
		sents[index].push_back(sentence);
		return;
	}
	else if ((ret = s.find(" is not guilty.")) != -1) {
		ret2 = s.rfind(' ', ret - 1);
		sentence.info[0] = name2index[s.substr(ret2 + 1, ret - 1)];
		sentence.info[1] = 0;
		sents[index].push_back(sentence);
		return;
	}
}

void init_week() {
	week2index.insert(pair<string, int>("Today is Monday.", 1));
	week2index.insert(pair<string, int>("Today is Tuesday.", 2));
	week2index.insert(pair<string, int>("Today is Wednesday.", 3));
	week2index.insert(pair<string, int>("Today is Thursday.", 4));
	week2index.insert(pair<string, int>("Today is Friday.", 5));
	week2index.insert(pair<string, int>("Today is Saturday.", 6));
	week2index.insert(pair<string, int>("Today is Sunday.", 7));
}

int main() {
	init_week();
	cin >> m >> n >> p;
	string name;
	for (int i = 1; i <= m; i++) {
		cin >> name;
		name2index.insert(pair<string, int>(name, i));
	}
	for (int i = 1; i <= p; i++) {
		cin >> name;
		name = name.substr(0, name.size() - 1);
		get_sent(name2index[name]);
	}

	int cnt = 0, cnt2 = 0, find = 0;
	bool end_index = false;
	for (int guilty = 1; guilty <= m; guilty++) {//假设罪犯
		for (int day = 1; day <= 7; day++) {//假设日期
			cnt = 0;//说假话的人数
			cnt2 = 0;//说真话的人数
			memset(states + 1, -1, sizeof(int) * max_m);//初始化未知身份
			end_index = false;//开始推断
			for (int i = 1; i <= m && !end_index; i++) {//遍历每一个人
				for (int j = 0; j < int(sents[i].size()); j++) {//遍历这个人说过的话
					if (sents[i][j].info[0]) {//关于人
						if (sents[i][j].info[1]) {//xx是罪犯
							if (guilty != sents[i][j].info[0]) {//假话
								if (states[i] != -1) {//身份已知
									if (states[i] != 0) {
										end_index = true;//与已知矛盾,假设错误,本轮推断结束
										break;
									}
								}
								else {//身份未知
									states[i] = 0;//标记说假话的身份
									cnt++;//说假话人数++
								}
							}
							else {//真话
								if (states[i] != -1) {//身份已知
									if (states[i] != 1) {
										end_index = true;//与已知矛盾,假设错误,本轮推断结束
										break;
									}
								}
								else {//身份未知
									states[i] = 1;//标记说真话身份
									cnt2++;//说真话的人++
								}
							}
						}
						else {//xx不是罪犯
							if (guilty == sents[i][j].info[0]) {//假话
								if (states[i] != -1) {//身份已知
									if (states[i] != 0) {
										end_index = true;//与已知矛盾,假设错误,本轮推断结束
										break;
									}
								}
								else {//身份未知
									states[i] = 0;//标记说假话的身份
									cnt++;//说假话人数++
								}
							}
							else {//真话
								if (states[i] != -1) {//身份已知
									if (states[i] != 1) {
										end_index = true;//与已知矛盾,假设错误,本轮推断结束
										break;
									}
								}
								else {//身份未知
									states[i] = 1;//标记说真话的身份
									cnt2++;//说真话的人++
								}
							}
						}
					}
					else {//关于日期
						if (day != sents[i][j].info[1]) {//假话
							if (states[i] != -1) {//身份已知
								if (states[i] != 0) {
									end_index = true;//与已知矛盾,假设错误,本轮推断结束
									break;
								}
							}
							else {//身份未知
								states[i] = 0;//标记说假话的身份
								cnt++;//说假话人数++
							}
						}
						else {//真话
							if (states[i] != -1) {//身份已知
								if (states[i] != 1) {
									end_index = true;//与已知矛盾,假设错误,本轮推断结束
									break;
								}
							}
							else {//身份未知
								states[i] = 1;//标记说真话的身份
								cnt2++;//说真话的人++
							}
						}
					}
				}
			}//一轮推断完成
			if (cnt <= n && cnt2 <= m - n && !end_index) {//推断出凶手
				if (find == 0)
					find = guilty;
				else if (find != guilty) {//推断出多个凶手
					cout << "Cannot Determine" << endl;
					return 0;
				}
			}
		}
	}
	//结论
	if (find) {
		for (map<string, int>::iterator it = name2index.begin(); it != name2index.end(); it++) {
			if (it->second == find) {
				cout << it->first << endl;
			}
		}
	}
	else cout << "Impossible" << endl;//未推断出凶手
	return 0;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/393611.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

关于Ping命令的七种用法

今天我们来详细看下ping命令详细使用&#xff1a; 一、ping基本使用详解 在网络中ping是一个十分强大的TCP/IP工具。它的作用主要为&#xff1a; 1、用来检测网络的连通情况和分析网络速度 2、根据域名得到服务器IP 3、根据ping返回的TTL值来判断对方所使用的操作系统及数据…

前端——2.HTML基本结构标签

这篇文章我们从0来介绍一下HTML的相关标签内容 目录 1.HTML语法规范 1.1基本语法概述 1.2标签关系 2.HTML的基本结构标签 2.1第一个HTML网页 2.2基本结构标签总结 1.HTML语法规范 下面&#xff0c;我们来看一下HTML的语法规范的内容 1.1基本语法概述 首先&#xff0c…

基于粒子群优化支持向量机SVM发电功率回归分析,eemd-pso-svm

目录 支持向量机SVM的详细原理 SVM的定义 SVM理论 SVM应用实例,基于eemd分解+粒子群改进SVM的回归分析 代码 结果分析 展望 支持向量机SVM的详细原理 SVM的定义 支持向量机(support vector machines, SVM)是一种二分类模型,它的基本模型是定义在特征空间上的间隔最大的…

挖到宝了,这2款浏览器工具多看小说自由,没踩雷

浏览器除了可以用来搜索和工作&#xff0c;还有很多有趣的娱乐功能。例如&#xff0c;喜欢看片的朋友可以通过浏览器追剧看电影&#xff0c;小说爱好者可以通过浏览器看小说。那么&#xff0c;有没有哪些浏览器可以免费阅读小说呢&#xff1f;我挖到宝了&#xff0c;原来这2款浏…

分布式之CAP原则

文章目录一、知识储备1.1 一致性&#xff08;Consistency&#xff09;1.2 可用性&#xff08;Availability&#xff09;1.3 分区容错性&#xff08;Partition tolerance&#xff09;二、CAP原则2.1 证明三、常见分布式系统采用的原则3.1 CP原则3.2 AP原则3.3 CA原则3.4 动态调节…

可视化爬虫框架spiderflow入门及实战

官网: 点击直达官网 文档: 点击查看官网文档 以下内容部分来自官网或官网文档。文章比较长&#xff0c;请准备好瓜子和小板凳~~~ TIP&#xff1a; 文中用到的网站地址仅为了说明功能&#xff0c;如有侵犯&#xff0c;请告知&#xff0c;会及时删除或者修改本文仅供学习参考&am…

合作伙伴管理软件VS CRM,企业应该选择哪一个?

当涉及到管理你公司的伙伴关系和与客户的关系时&#xff0c;有两个主要选择&#xff1a;合作伙伴管理软件和CRM&#xff08;客户关系管理&#xff09;软件。虽然这两种工具都可以帮助你跟踪商业关系的重要信息&#xff0c;但它们都有各自的优势和不足。 合作伙伴管理软件是专门…

没有钢琴也可实现弹奏自由?实时在Jetson上运行单阶段手指关键点模型

钢琴是人类创作音乐的经典乐器&#xff0c;程序是实现创意的工具之魂。今天我给大家分享用程序实现的桌上钢琴师项目。本项目基于飞桨实现一个虚拟钢琴&#xff0c;让大家可以在任意平面上弹奏钢琴&#xff0c;实现弹奏自由。 该项目的原理是利用手部关键点检测模型识别手的关键…

金三银四,我猜你需要这套网络安全工程师面试题合集【全网首发】

以下为信息安全/网络安全各个方向涉及的面试题&#xff0c;星数越多代表问题出现的几率越大&#xff0c;祝各位都能找到满意的工作~ 注&#xff1a;做这个List的目标不是很全&#xff0c;因为无论如何都不可能覆盖所有的面试问题&#xff0c;更多的还是希望由点达面&#xff0…

h264编码之SPS解析

一、概念 SPS即Sequence Paramater Set&#xff0c;又称作序列参数集。SPS中保存了一组编码视频序列(Coded video sequence)的全局参数。 二、定义 H.264标准协议中规定的SPS格式位于文档的7.3.2.1.1&#xff0c;如下图所示&#xff1a; 1、profile_idc 根据《T-REC-H.264-2…

数据库的备份与恢复

文章目录前言备份恢复概述故障的种类数据库备份数据库的恢复日志文件前言 提示&#xff1a;这里可以添加本文要记录的大概内容&#xff1a; 备份与恢复是为了防止数据库运行出现故障时造成数据丢失、损坏的一个重要手段 提示&#xff1a;以下是本篇文章正文内容&#xff0c;…

【SAP PO】X-DOC:SAP PO 接口配置完整教程之三REST服务发布报文转换

X-DOC&#xff1a;SAP PO 接口配置完整教程之三REST服务发布&报文转换1、SAP端RFC开发2、PO端接口配置2.1、PO端SLD配置2.2、PO端ESB配置2.3、PO端IB配置2.4、Postman测试3、报文转换配置1、SAP端RFC开发 开发一个支持远程访问的RFC&#xff1a; FUNCTION yfm_xl_rest_de…

水库大坝安全监测及水情自动测报系统(设备清单)

解决方案水库大坝安全监测及水情自动测报系统方案&#xff0c;系统主要由降雨量监测站、水库水位监测站、大坝安全监测中的渗流量、渗流压力和变形监测站及视频和图像监测站等站点组成&#xff0c;同时建立规范、统一的监测平台&#xff0c;集数据传输、信息共享、数据储存于一…

springBoot健康检查与k8s探针

从springBoot健康检查到k8s探针的问题定位 环境&#xff1a; k8s、springboot、mysql、flyway、Spring boot 2.0 Actuator健康检查组件等 问题&#xff1a; 项目中集成了flyway&#xff0c;当项目第一次启动时&#xff0c;都会初始化flyway文件&#xff0c;再加上各种组件的…

水文模型有哪些?SWAT模型、VIC模型、HEC模型、HSPF模型、HYPE模型、SWMM模型、FVCOM模型、Delft3D模型等应用

目录 ㈠ 从小白到精通SWAT模型学习建模方法、实例应用、高级进阶 ㈡ RVIC模型融合实践技术应用及未来气候变化模型预测 ㈢ HEC-RAS一维、二维建模方法及实践技术应用 ㈣ HEC-HMS水文模型实践技术应用 ㈤ HSPF 模型应用 ㈥ HYPE分布式水文模型建模方法与案例分析 ㈦ 基于…

raspberry安装ffmpeg以及扩展了解

文章目录ffmpeg简介使用ffmpegffplay播放控制命令播放滤镜处理参数选项ffprobeffmpeg sudo apt-get install ffmpeg简介 FFmpeg的名称来自MPEG视频编码标准&#xff0c;前面的“FF”代表“Fast Forward”&#xff0c;FFmpeg是一套可以用来记录、转换数字音频、视频&#xff0…

Godot Engine 4.0横空出世,Vulkan大怪兽加持,画质提升简直亮瞎眼

【CSDN 编者按】经历了漫长的等待&#xff0c;万众瞩目的 Godot Engine 4.0 正式版在其 3.0 版本发布 5 年以后&#xff0c;终于带着海量令人兴奋的新功能横空出世&#xff01; 整理 | 开发游戏的老王 责编 | 王子彧 出品 | CSDN&#xff08;ID&#xff1a;CSDNnews&#xff09…

优思学院|盘点,精益生产25个工具!【必需收藏】

精益生产方法需要一种全面的方法才能有效实施。精益这个概念是每个接触产品供应链的人都要实践的&#xff0c;无论是在计划方面还是在分析方面。 精益生产工具有助于持续改进生产效率和产品或服务质量。精益工具是要减少 Muda &#xff08;浪费&#xff09;&#xff0c;从生产过…

内存清理、动画制作、CPU检测等五款实用软件推荐

人类与99%的动物之间最大差别在于是否会运用工具&#xff0c;借助好的工具&#xff0c;能提升几倍的工作效率。 1.内存清理软件——MemReduct MemReduct是一款内存清理软件&#xff0c;现在越来越多的软件由于硬件的普遍发展&#xff0c;对内存的使用都开始肆无忌惮起来&…

【C++】list的使用和基本迭代器框架的实现 vs和g++下string结构的说明

真正的成熟应该并不是追求完美&#xff0c;而是直面自己的缺憾&#xff0c;这才是生活的本质。 文章目录一、初见list1.list的迭代器失效和基本使用2.list的operations操作接口&#xff08;看起来挺不错的接口&#xff0c;但可惜不怎么实用&#xff09;3.vector和list的排序性能…