POJ 1478 / UVA592-Island of Logic -有点繁琐的暴力枚举+模拟

稍稍情况有点麻烦。。。主要是代码写得挫了

给出n句话

让你判断 有什么信息是确定的 ,例如【每个人的身份是否确定】、【当前时间是否确定】

身份有三种,神、人、恶魔,神只说真话,恶魔只说假话,人在白天说真话,在晚上说假话

 

n句话 里 有肯定句和否定句


可以把每句话 记录到一个结构体

每句话记录说话者

   1、是否只叙述时间 如【It is ( day | night ). 】

   2、是否lying句式, 如【I am [not] lying】,记下肯定还是否定

  3、身份句式【X is [not] 身份】  记下 被叙述者,以及身份,以及肯定还是否定

因为人数最多只有5种类型,时间只有2种,我们直接枚举所有的身份情况+时间情况,也就是【 3^5 * 2 】

每一种枚举下

我们先  判断本句话是否为真,

再判断说话者 是谁 即可


特判一下impossible的情况  【i am lying】


最后如果方案唯一,则 输出答案

如果合法方案有多种,我们看 在多种方案中,是否有 身份一直不变的人, 是否 时间一直不变,如果有,则它们是确定的。

如果没,输出 【No facts are deducible.】


以下数据是从网上找到的 

----------------------------------------------------

6
A: B is human.
A: B is evil.
B: A is human.
C: A is not lying.
B: C is not human.
D: E is not lying.
4
A: I am human.
A: It is night.
B: I am human.
B: It is day.
3
A: I am human.
B: I am human.
A: B is lying.
3
A: I am divine.
B: A is not lying.
A: B is lying.
3
A: I am divine.
B: A is lying.
A: B is lying.
5
A: B is human.
A: B is evil.
B: A is evil.
C: A is not lying.
B: It is day.
5
C: A is not lying.
A: B is human.
A: B is evil.
B: A is evil.
B: It is day.
1
A: A is not lying.
1
A: A is lying.
2
E: E is evil.
E: E is divine.
7
A: It is night.
B: It is day.
C: I am human.
E: C is human.
C: E is divine.
A: B is lying.
B: C is evil.
0

*********************************************************************************

Conversation #1
A is human.
B is divine.
C is evil.
It is night.

Conversation #2
A is evil.
B is human.
It is day.

Conversation #3
It is day.

Conversation #4
This is impossible.

Conversation #5
No facts are deducible.

Conversation #6
A is evil.
B is divine.
C is evil.
It is day.

Conversation #7
A is evil.
B is divine.

C is evil.
It is day.

Conversation #8
No facts are deducible.

Conversation #9
This is impossible.

Conversation #10
E is human.
It is night.

Conversation #11
A is evil.
C is evil.
E is evil.
It is day.

********************************************************************************



#include <cstdio>
#include <cmath>
#include <cstring>
#include <string>
#include <algorithm>
#include <iostream>
#include <queue>
#include <map>
#include <set>
#include <vector>
using namespace std; 
struct node
{
	char spker;
	char who;
	int is_not;
	string what;
	int d_or_n;
};
node tm[55];

string ss;  

vector<int> sb[55];
set<char> ::iterator it;
int main()
{
	
	int n,i;
	int cnt=1;
	set<char> sb;
	while(scanf("%d",&n)!=EOF&&n)
	{
		sb.clear();
		getchar();
		for (i=1;i<=n;i++)
		{
			getline(cin,ss);
			string sub=ss.substr(3,ss.length()-3);		//截取除去主语后的一句话
			tm[i].d_or_n=-1;			// 为-1表示本句话与时间无关
			if (sub=="It is day.")
			{  
				tm[i].spker=ss[0];				//记下说话人
				tm[i].d_or_n=1;continue;		//记下是白天还是黑夜
			}
			if (sub=="It is night.")
			{  
				tm[i].spker=ss[0];
				tm[i].d_or_n=0;continue;
			} 
			if (ss[3]=='I')			//‘I’就是说话人自己
				ss[3]=ss[0];
			tm[i].spker=ss[0];
			tm[i].who=ss[3];
			sb.insert(tm[i].spker);		//计算种类
			sb.insert(tm[i].who);
			if (ss.find("not")!=string::npos)	//如果找不到not,则为肯定句,找到则表示为否定句
				tm[i].is_not=0;
			else
				tm[i].is_not=1;
			if (ss.find("divine")!=string::npos)		//被认为是什么
				tm[i].what="divine";
			else	if (ss.find("human")!=string::npos)
				tm[i].what="human";
			else	if (ss.find("evil")!=string::npos)
				tm[i].what="evil";
			else	if (ss.find("lying")!=string::npos)
				tm[i].what="lying"; 
		}
		
		printf("Conversation #%d\n",cnt++);		
		
		string what[6];		//what[i]存第i个人 当前的身份
		int k;
		int flag=0;			// impossible的情况(当且仅当说I am lying.才是此情况)
		int kind=sb.size();		//出现的种类数(最后要输出的种类数)
		int possible_way=0;		//合法的身份方案,  如果仅有一种,则输出,否则表示都多种情况 
		int all=(int)pow(3.0,5);	//枚举范围0-all
		string ans[6];				//用来保存 找到合法的身份方案后 的那5个身份 
		int has_change[6];			//用来记录 找到的多个合法方案中 某人的身份是否一样 (如果一样,表示某个人的身份在什么时候都是确定的,反之则否)
		memset(has_change,0,sizeof(has_change));
		int ans_d_n=-1;	//用来保存 找到合法的身份方案后 的时间,如果一直不变,表示这个时间也是固定的,同has_change数组的理
		int has_changed_d_n=0;	//记录 合法方案的时间是否改变,理由同上
		int line=1;			//标志 当前方案是否第一次出现,
		int day_night;		//时间变量,1为白天,0为黑夜
		int twice=0;		//枚举 每种方案 在白天 和 黑夜下的情况
		for (k=0;k<all;k++)
		{
			
			if (twice==1)//第二次 把时间看作黑夜,并且k--回到上个方案
			{
				twice++;
				k--;
				day_night=0;
				continue;
			}
			
			if (twice==0)//第一次 把时间看作白天
			{
				twice++;
				day_night=1;
			} 
			if (twice==2)		//复位
				twice=0;
			
			int tmp=k; 
			
			for (i=1;i<=5;i++)		//枚举身份
			{
				
				if (tmp%3==0) 
					what[i]="divine";
				else
					if (tmp%3==1) what[i]="human";
					else
						what[i]="evil";
					tmp=tmp/3;
			}
			
			int find_ans=1;		//是否找到一个合法方案
			for (i=1;i<=n;i++)
			{
				if (tm[i].d_or_n!=-1)	//如果第i句话与时间相关
				{
					
					if (day_night==tm[i].d_or_n)  //如果第i句话时间与当前方案时间相同,即本句话为真 
					{
						if (what[tm[i].spker-'A'+1]=="divine") //神族说真话,没问题
							continue;
						if (what[tm[i].spker-'A'+1]=="evil")  //恶魔说真话,本方案不合法,结束
						{find_ans=0;break;}
						if (what[tm[i].spker-'A'+1]=="human")	//人
						{
							if (day_night==1) continue;		//白天说真话,ok
							else 	{find_ans=0;break;}		//晚上说真话,本方案不合法,结束
						} 
					}
					else	 //如果第i句话时间与当前方案时间相同,即本句话为假 ,下面的具体就不分析了
					{
						if (what[tm[i].spker-'A'+1]=="divine")	
						{find_ans=0;break;}
						if (what[tm[i].spker-'A'+1]=="evil")
							continue;
						if (what[tm[i].spker-'A'+1]=="human")
						{
							if (day_night==1) {find_ans=0;break;}
							else 	continue;
						} 
					}
				}
				//下面的都是本句话与时间 无关的情况
				
				if (tm[i].what=="lying")		//谁谁谁 lying 的情况
				{
					if (tm[i].is_not==1)		//如果说 某人【在】说谎
					{
						if (tm[i].spker==tm[i].who)  //如果自己说自己说谎, 这是唯一的 不可能的方案,flag标记,结束所有方案,输出答案
						{flag=1;	find_ans=1;break;}
						
						int isreal;						//其余情况要先判断 这句话是否为真 (先不考虑说话人,只考虑说的话)
						if (what[tm[i].who-'A'+1]=="divine")	 //神不可能撒谎,本句话为假
							isreal=0;
						if (what[tm[i].who-'A'+1]=="human")		//人类撒谎根据时间
						{ 
							if (day_night==1)	isreal=0;
							else isreal=1;
						} 
						if (what[tm[i].who-'A'+1]=="evil")		//恶魔撒谎
							isreal=1;
						
						if (isreal)  	//如果 shuo 的是真话
						{
							if (what[tm[i].spker-'A'+1]=="divine")	//神说真话,合法
								continue; 
							if (what[tm[i].spker-'A'+1]=="human")	//人是否合法得看时间
							{ 
								if (day_night==0)
								{ find_ans=0;break; }
							}
							if (what[tm[i].spker-'A'+1]=="evil")	//恶魔不说真话
							{ find_ans=0;break; } 
						}
						else	//说的是假话
						{
							if (what[tm[i].spker-'A'+1]=="divine")
							{find_ans=0;break;} //矛盾
							if (what[tm[i].spker-'A'+1]=="human")
							{
								if (day_night==1)	{find_ans=0;break;} //本应该说真话却说假话
								else if (day_night==0) continue; //晚上说假话 
							}
							if (what[tm[i].spker-'A'+1]=="evil")
								continue; 
						}
					}	
					else		//	//如果说 某人【不在】说谎
					{
						int isreal;								//依旧先判断 话的真假
						if (what[tm[i].who-'A'+1]=="divine")
							isreal=1;
						if (what[tm[i].who-'A'+1]=="human")
						{ 
							if (day_night==1)	isreal=1;
							else isreal=0;
						} 
						if (what[tm[i].who-'A'+1]=="evil")
							isreal=0;
						
						
						if (isreal)	//如果 shuo 的是真话
						{
							if (what[tm[i].spker-'A'+1]=="divine")
								continue; 
							if (what[tm[i].spker-'A'+1]=="human")
							{ 
								if (day_night==0)
								{ find_ans=0;break; }
							}
							if (what[tm[i].spker-'A'+1]=="evil")
							{ find_ans=0;break; }
						}
						else		//如果说的是假话
						{
							if (what[tm[i].spker-'A'+1]=="divine")
							{find_ans=0;break;} //矛盾
							if (what[tm[i].spker-'A'+1]=="human")
							{
								if (day_night==1)	{find_ans=0;break;} //本应该说真话却说假话
								else if (day_night==0) continue; //晚上说假话 
							}
							if (what[tm[i].spker-'A'+1]=="evil")
								continue;
							
						}
					}
				}
				else  	//下面的情况 就是 xxx是什么身份 这种句式
				{
					if (tm[i].is_not==1)     //如果是肯定句
					{
						if (tm[i].what==what[tm[i].who-'A'+1])//与被描述者身份一致,本句为真话  
						{
							if (what[tm[i].spker-'A'+1]=="divine")
								continue;
							if (what[tm[i].spker-'A'+1]=="human")
							{
								if (day_night==1) continue;  //本应该说真话却说假话
								else if (day_night==0) 	{find_ans=0;break;} //晚上说假话 
							}
							if (what[tm[i].spker-'A'+1]=="evil")
							{find_ans=0;break;}
						}
						else //与被描述者身份不一致,本句为假话  
						{
							if (what[tm[i].spker-'A'+1]=="divine")
							{find_ans=0;break;} //矛盾
							if (what[tm[i].spker-'A'+1]=="human")
							{
								if (day_night==1)	{find_ans=0;break;} //本应该说真话却说假话
								else if (day_night==0) continue; //晚上说假话 
							}
							if (what[tm[i].spker-'A'+1]=="evil")
								continue;
						}
					}
					else			//如果整句话是否定句 
					{
						if (tm[i].what==what[tm[i].who-'A'+1])   //与被描述者身份一致,本句为假话  
						{
							if (what[tm[i].spker-'A'+1]=="divine")
							{find_ans=0;break;} //矛盾
							if (what[tm[i].spker-'A'+1]=="human")
							{
								if (day_night==1) 	{find_ans=0;break;}  //本应该说真话却说假话
								else if (day_night==0) continue; //晚上说假话 
							}
							if (what[tm[i].spker-'A'+1]=="evil")
								continue;
						}
						else					//与被描述者身份不一致,本句为真话  
						{
							if (what[tm[i].spker-'A'+1]=="divine")
								continue;
							if (what[tm[i].spker-'A'+1]=="human")
							{
								if (day_night==1) continue;  //本应该说真话却说假话
								else if (day_night==0) 	{find_ans=0;break;}//晚上说假话 
							}
							if (what[tm[i].spker-'A'+1]=="evil")
							{find_ans=0;break;}
						}
					}
				}
			}
			
			if (flag)		//impossible的情况
				break; 
			if (find_ans)		//找到一种合法方案
			{	 
				int h;
				for (h=1;h<=5;h++)		//储存答案
				{ 
					if (line)		//如果是第一次找到合法方案,直接存下来,并在后面去掉首次标记
					{
						ans[h]=what[h];continue;   
					}
					if (ans[h]!=what[h])		//非第一次找到答案,先比较当前答案与以前的是否一致
					{
						ans[h]=what[h];			//更新答案并且
						has_change[h]=1;		//如果不一致,标志这个人的身份是不确定的
					}
				}
				if (ans_d_n!=day_night&&!line)	//更新时间,并看是否一直不变
					has_changed_d_n=1;
				ans_d_n=day_night;		//储存时间
				possible_way++;			//方案数++ 
				line=0;
			}	
	}
	
	
	if (possible_way==0||flag)   //无合法方案,或者遇到flag情况
		printf("This is impossible.\n");
	else
		if (possible_way>1)		//方案有很多
		{
			int has=0;		//记录是否有输出过答案
			it=sb.begin();		//遍历出现过的人
			for (int h=1;h<=kind;h++,it++)	
			{
				int p= (*it)-'A'+1;		//出现过的人的编号
				if (has_change[p]==0)		//如果在多个方案中 身份都一样,则标明身份确定,输出,并更新标记
				{
					printf("%c is %s.\n",p-1+'A',ans[p].c_str());
					has=1;							//表示输出过答案
				}
			}
			
			if (has_changed_d_n==0)		//如果时间在多个方案中不曾改变
				if (ans_d_n!=-1)		
					printf("It is %s.\n",(ans_d_n==1)?"day":"night"),has=1;		//输出答案并更新标记
				
				if (!has)			//如果 人的身份或时间都不是确定的,输出 无法确定事实
					printf("No facts are deducible.\n");
		}
		else		//如果方案只有一种。。。。必然就是合法的唯一的方案
		{
			it=sb.begin();
			
			for (i=1;i<=kind;i++,it++)
			{
				int p= (*it)-'A'+1;
				printf("%c is %s.\n",p-1+'A',ans[p].c_str());
			}
			if (ans_d_n!=-1)							
				printf("It is %s.\n",(ans_d_n==1)?"day":"night");
			
		} 
		printf("\n");		
		}  
		
		
		return 0;
}


你可能感兴趣的:(POJ 1478 / UVA592-Island of Logic -有点繁琐的暴力枚举+模拟)