[洛谷P2482][SDOI2010]猪国杀

题目大意:猪国杀,又一道大模拟题

题解:模拟,对于一个没有玩过三国杀的人来说,一堆细节不知道,写的十分吃力

卡点:无数,不想说什么了,这告诉我要多玩游戏

 

C++ Code:

#include <algorithm>
#include <cstdlib>
#include <cstdio>
#include <iostream>
#include <map>
#include <queue>
#include <vector>
//#define DEBUG

#define Main_Pig 1
using std::string;

namespace Debug {
	std::map<string, string> Name;
	inline void init() {
		Name[""] = "无"; 
		Name["P"] = "桃"; 
		Name["K"] = "杀"; 
		Name["D"] = "闪"; 
		Name["F"] = "决斗"; 
		Name["N"] = "南猪入侵"; 
		Name["W"] = "万箭齐发"; 
		Name["J"] = "无懈可击"; 
		Name["Z"] = "猪哥连弩"; 
		Name["MP"] = "主猪"; 
		Name["ZP"] = "忠猪"; 
		Name["FP"] = "反猪"; 
	}
}

const string NONE = "";
//基本牌 
const string PEACH = "P",
			 KILL = "K",
			 DODGE = "D";

//锦囊牌 
const string JD = "F",
			 NZRQ = "N",
			 WJQF = "W",
			 WXKJ = "J";

//装备牌
const string ZGLN = "Z";

//猪 
const string MP = "MP",
			 ZP = "ZP",
			 FP = "FP";

//主猪,跳忠,不知,类反猪,跳反 
const int TM = 2,
		  TZ = 1,
		  UK = 0,
		  Similar_FZ = -1,
		  TF = -2; 


int Pigs_Num;

struct Card {
	std::string type;
	inline Card(string __type = NONE) {type = __type;}
	
	friend std::istream & operator >> (std::istream &Fin, Card &card) {
		string __card;
		Fin >> __card;
		card = Card(__card);
		return Fin;
	}
	friend std::ostream & operator << (std::ostream &Fout, Card &card) {
		string __type = card.type;
		#ifdef DEBUG
		Fout << Debug::Name[__type];
		#else
		Fout << __type;
		#endif
		return Fout;
	}
	
	inline bool is_kill() {return type == KILL;}
	inline bool is_peach() {return type == PEACH;}
	inline bool is_dodge() {return type == DODGE;}
	inline bool is_jd() {return type == JD;}
	inline bool is_nzrq() {return type == NZRQ;}
	inline bool is_wjqf() {return type == WJQF;}
	inline bool is_wxkj() {return type == WXKJ;}
	inline bool is_zgln() {return type == ZGLN;}
	
};

struct Card_Pool {
	std::queue<Card> cards;
	Card get_card(bool Empty = false) { //抽牌 
		if (cards.empty()) return NONE;
		Card __card = cards.front();
		if (cards.size() > 1 || Empty) cards.pop();
		return __card;
	}
	bool empty() {
		return cards.empty();
	}
	
	void insert(string __card) {cards.push(Card(__card));} //牌堆加牌 
	void insert(Card __card) {cards.push(__card);} //牌堆加牌 
	
	friend std::ostream & operator << (std::ostream &Fout, Card_Pool &__Pool) { //从上到下输出牌堆的牌 
		std::queue<Card> tmp;
		while (!__Pool.empty()) {
			Card __card = __Pool.get_card(true);
			tmp.push(__card);
			#ifdef DEBUG
			Fout << Debug::Name[__card.type];
			#else
			Fout << __card.type;
			#endif
		}
		while (!tmp.empty()) {
			__Pool.insert(tmp.front());
			tmp.pop();
		}
		return Fout;
	}
	
} Pool;

struct Pig {
	string type;
	int Jump, Max_HP, HP, Pos;
	bool AK_47, dead;
	std::vector<Card> cards;
	
	inline std::vector<Card>::iterator begin() {return cards.begin();}
	inline std::vector<Card>::iterator end() {return cards.end();}
	
	inline Pig(string __type = NONE, int __pos = 0) {
		Pos = __pos;
		type = __type;
		if (__type == MP) Jump = TM;
		else Jump = UK;
		Max_HP = HP = 4;
		AK_47 = dead = false;
		cards.clear();
	}
	
	inline void clear() {
		cards.clear();
		AK_47 = false;
	}
	
	void insert_zgln(std::vector<Card>::iterator it) {
		cards.erase(it);
		AK_47 = true;
	}
	void insert(string __card) { //把一张牌加入手牌
		cards.push_back(Card(__card));
	}
	void insert(Card __card) { //把一张牌加入手牌 
		cards.push_back(__card);
	}
	
	inline bool max_hp() {return Max_HP <= HP;} //是否满血 
	
	friend std::ostream & operator << (std::ostream &Fout, Pig pig) {
		if (pig.dead) {
			Fout << "DEAD";
			return Fout;
		}
		for (std::vector<Card>::iterator it = pig.begin(); it != pig.end(); it++) {
			Fout << *it << " ";
		}
		return Fout;
	}
	
	std::vector<Card>::iterator find_peach() { //找桃,并返回位置 
		for (std::vector<Card>::iterator it = cards.begin(); it != cards.end(); it++) {
			if (it -> is_peach()) return it;
		}
		return cards.end();
	}
	bool use_peach() { //使用桃 
		if (max_hp()) return false;
		std::vector<Card>::iterator it = find_peach();
		if (it == cards.end()) return false;
		HP++;
		cards.erase(it);
		#ifdef DEBUG
		std::cout << std::endl;
		std::cout << type << ":" << Pos << "用了一个桃,现在血量为" << HP << std::endl;
		std::cout << "现在拥有的牌 :" << *this << std::endl;
		#endif 
		return true;
	}
	
	std::vector<Card>::iterator find_kill() { //找杀,并返回位置 
		for (std::vector<Card>::iterator it = cards.begin(); it != cards.end(); it++) {
			if (it -> is_kill()) return it;
		}
		return cards.end();
	}
	bool use_kill() { //使用闪 
		std::vector<Card>::iterator it = find_kill();
		if (it == cards.end()) return false;
		cards.erase(it);
		#ifdef DEBUG
		std::cout << std::endl;
		std::cout << type << ":" << Pos << "用了一个杀" << std::endl;
		std::cout << "现在拥有的牌 :" << *this << std::endl;
		#endif 
		return true;
	}
	
	std::vector<Card>::iterator find_dodge() { //找闪,并返回位置 
		for (std::vector<Card>::iterator it = cards.begin(); it != cards.end(); it++) {
			if (it -> is_dodge()) return it;
		}
		return cards.end();
	}
	bool use_dodge() { //使用闪 
		std::vector<Card>::iterator it = find_dodge();
		if (it == cards.end()) return false;
		cards.erase(it);
		#ifdef DEBUG
		std::cout << std::endl;
		std::cout << type << ":" << Pos << "用了一个闪" << std::endl;
		std::cout << "现在拥有的牌 :" << *this << std::endl;
		#endif 
		return true;
	}
	
	
	std::vector<Card>::iterator find_wxkj() { //找无懈可击,并返回位置 
		for (std::vector<Card>::iterator it = cards.begin(); it != cards.end(); it++) {
			if (it -> is_wxkj()) return it;
		}
		return cards.end();
	}
	bool use_wxkj() { //使用无懈可击
		std::vector<Card>::iterator it = find_wxkj();
		if (it == cards.end()) return false;
		cards.erase(it);
		#ifdef DEBUG
		std::cout << std::endl;
		std::cout << type << ":" << Pos << "用了无懈可击" << std::endl;
		std::cout << "现在拥有的牌 :" << *this << std::endl;
		#endif 
		return true;
	}
	
	
	
	bool hurt(int num = 1) { //收到伤害 
		HP -= num;
		#ifdef DEBUG
		std::cout << std::endl;
		std::cout << type << ":" << Pos << "收到" << num << "点伤害,现在血量为" << HP << std::endl;
		#endif 
		if (HP <= 0) {
			#ifdef DEBUG
			std::cout << type << ":" << Pos << "进入濒死状态" << std::endl;
			#endif 
			if (!use_peach()) {
				this -> clear();
				#ifdef DEBUG
				std::cout << type << ":" << Pos << "死亡" << std::endl;
				#endif 
				dead = true;
				return true;
			} 
		}
		return false;
	} 
	
	
} Pigs[20];




int nxt_pig(int now) { //找下一头猪 
	int nxt = now;
	while (true) {
		nxt = nxt % Pigs_Num + 1;
		if (!Pigs[nxt].dead) return nxt;
	}
	return 0;
}


int nxt_pig(Pig now) { //找下一头猪 
	return nxt_pig(now.Pos);
}

inline int game_over() { //判断游戏是否结束 
	if (Pigs[1].dead) return -1;
	int FP_Num = 0;
	for (int i = 1; i <= Pigs_Num; i++) {
		#ifdef DEBUG
		std::cout << std::endl;
		std::cout << "第" << i << "只猪是" << Pigs[i].type << ",是否死亡:" << Pigs[i].dead << std::endl; 
		#endif
		if (!Pigs[i].dead) FP_Num += (Pigs[i].type == FP);
	}
	#ifdef DEBUG
	std::cout << "现在反猪有" << FP_Num << "只" << std::endl; 
	#endif
	return !FP_Num;
}

void end_game() { //输出 
	int situation = game_over();
	if (situation == 1) puts("MP");
	else puts("FP");
	for (int i = 1; i <= Pigs_Num; i++) std::cout << Pigs[i] << std::endl;
	exit(0);
}


namespace Action {
	void result(int From, int To) { //奖励与惩罚(主猪杀忠猪,杀反猪) 
		#ifdef DEBUG
		std::cout << std::endl;
		std::cout << Pigs[From].type << ":" << Pigs[From].Pos << "杀死了" << Pigs[To].type << ":" << Pigs[To].Pos << std::endl;
		#endif
		if (game_over()) end_game();
		
		if (Pigs[To].type == FP) {
			for (int i = 0; i < 3; i++) Pigs[From].insert(Pool.get_card());
			#ifdef DEBUG
			std::cout << std::endl;
			std::cout << Pigs[From].type << ":" << Pigs[From].Pos << "杀死反猪,获得三张牌" << std::endl;
			std::cout << "现在的牌:" << Pigs[From] << std::endl;
			#endif
		}
		if (Pigs[To].type == ZP) {
			if (Pigs[From].type == MP) {
				Pigs[From].clear();
				#ifdef DEBUG
				std::cout << std::endl;
				std::cout << "主猪杀死忠猪丢弃所有牌" << std::endl;
				#endif
			}
		}
	}
	
	void enemies(int From, int To, bool aoe = false) { //from对to表敌意,是否是范围伤害 
		#ifdef DEBUG
		std::cout << std::endl;
		std::cout << Pigs[From].type << ":" << Pigs[From].Pos << "对" << Pigs[To].type << ":" << Pigs[To].Pos << "表敌意" << std::endl;
		if (aoe) std::cout << "是范围伤害" << std::endl; 
		#endif
		if (aoe) {
			if (Pigs[From].Jump == UK) {
				if (Pigs[To].Jump == TM) {
					Pigs[From].Jump = Similar_FZ;
					#ifdef DEBUG
					std::cout << Pigs[From].type << ":" << Pigs[From].Pos << "类反猪" << std::endl;
					#endif
					return ;
				}
			}
			return ;
		} 
		if (Pigs[From].Jump == UK || Pigs[From].Jump == Similar_FZ) {
			if (Pigs[To].Jump == TF) {
				Pigs[From].Jump = TZ;
				#ifdef DEBUG
				std::cout << Pigs[From].type << ":" << Pigs[From].Pos << "跳忠" << std::endl;
				#endif
				return ;
			}
			
			if (Pigs[To].Jump > UK) {
				Pigs[From].Jump = TF;
				#ifdef DEBUG
				std::cout << Pigs[From].type << ":" << Pigs[From].Pos << "跳反" << std::endl;
				#endif
				return ;
			}
		}
	} 
	
	void friends(int From, int To) { //from对to献殷勤 
		#ifdef DEBUG
		std::cout << std::endl;
		std::cout << Pigs[From].type << ":" << Pigs[From].Pos << "对" << Pigs[To].type << ":" << Pigs[To].Pos << "献殷勤" << std::endl;
		#endif
		if (Pigs[From].Jump == UK || Pigs[From].Jump == Similar_FZ) {
			if (Pigs[To].Jump == TF) {
				Pigs[From].Jump = TF;
				#ifdef DEBUG
				std::cout << Pigs[From].type << ":" << Pigs[From].Pos << "跳反" << std::endl;
				#endif
				return ;
			}
			
			if (Pigs[To].Jump > UK) {
				Pigs[From].Jump = TZ;
				#ifdef DEBUG
				std::cout << Pigs[From].type << ":" << Pigs[From].Pos << "跳忠" << std::endl;
				#endif
				return ;
			}
			
		}
	} 
	
	bool hurt(int From, int To, bool aoe = false, bool jd = false) { //from对to造成伤害,是否是范围伤害,是否是决斗 
		#ifdef DEBUG
		std::cout << std::endl;
		std::cout << Pigs[From].type << ":" << Pigs[From].Pos << "对" << Pigs[To].type << ":" << Pigs[To].Pos << "造成伤害" << std::endl;
		#endif
		if (jd || aoe || !Pigs[To].use_dodge()) {
			if (Pigs[To].hurt()) result(From, To);
		}
		if (!jd) enemies(From, To, aoe);
		return false;
	}
	
	void kill(int From, int To, std::vector<Card>::iterator it) { //杀 
		Pigs[From].cards.erase(it);
		#ifdef DEBUG
		std::cout << std::endl;
		std::cout << Pigs[From].type << ":" << Pigs[From].Pos << "对" << Pigs[To].type << ":" << Pigs[To].Pos << "使用杀" << std::endl;
		#endif
		hurt(From, To); 
	}
	
	
	bool query_wxkj(int start, int Now, bool friends) { //询问是否使用无懈可击,开始的猪,目标猪,是否献殷勤 
		#ifdef DEBUG
		std::cout << std::endl;
		std::cout << "询问无懈可击,对象:" << Pigs[Now].type << ":" << Pigs[Now].Pos << std::endl;
		#endif
		if (friends) {
			int i = start;
			do {
				if (Pigs[i].type == ZP) {
					if (Pigs[Now].Jump > UK) {
						if (Pigs[i].use_wxkj()) {
							#ifdef DEBUG
							std::cout << std::endl;
							std::cout << Pigs[i].type << ":" << Pigs[i].Pos << "对" << Pigs[Now].type << ":" << Pigs[Now].Pos << "使用无懈可击" << std::endl;
							#endif
							Action::friends(i, Now);
							return !query_wxkj(i, i, false);
						}
					}
				}
				if (Pigs[i].type == MP) {
					if (Pigs[Now].Jump > UK) {
						if (Pigs[i].use_wxkj()) {
							#ifdef DEBUG
							std::cout << std::endl;
							std::cout << Pigs[i].type << ":" << Pigs[i].Pos << "对" << Pigs[Now].type << ":" << Pigs[Now].Pos << "使用无懈可击" << std::endl;
							#endif
							Action::friends(i, Now);
							return !query_wxkj(i, i, false);
						}
					}
				}
				if (Pigs[i].type == FP) {
					if (Pigs[Now].Jump == TF) {
						if (Pigs[i].use_wxkj()) {
							#ifdef DEBUG
							std::cout << std::endl;
							std::cout << Pigs[i].type << ":" << Pigs[i].Pos << "对" << Pigs[Now].type << ":" << Pigs[Now].Pos << "使用无懈可击" << std::endl;
							#endif
							Action::friends(i, Now);
							return !query_wxkj(i, i, false);
						}
					}
				}
				i = nxt_pig(i);
			} while (i != start);
			return false;
		} else {
			int i = start;
			do {
				if (Pigs[i].type == ZP) {
					if (Pigs[Now].Jump == TF) {
						if (Pigs[i].use_wxkj()) {
							#ifdef DEBUG
							std::cout << std::endl;
							std::cout << Pigs[i].type << ":" << Pigs[i].Pos << "对" << Pigs[Now].type << ":" << Pigs[Now].Pos << "使用无懈可击" << std::endl;
							#endif
							enemies(i, Now);
							return !query_wxkj(i, i, false);
						}
					}
				}
				if (Pigs[i].type == MP) {
					if (Pigs[Now].Jump < UK) {
						if (Pigs[i].use_wxkj()) {
							#ifdef DEBUG
							std::cout << std::endl;
							std::cout << Pigs[i].type << ":" << Pigs[i].Pos << "对" << Pigs[Now].type << ":" << Pigs[Now].Pos << "使用无懈可击" << std::endl;
							#endif
							enemies(i, Now);
							return !query_wxkj(i, i, false);
						}
					}
				}
				if (Pigs[i].type == FP) {
					if (Pigs[Now].Jump > UK) {
						if (Pigs[i].use_wxkj()) {
							#ifdef DEBUG
							std::cout << std::endl;
							std::cout << Pigs[i].type << ":" << Pigs[i].Pos << "对" << Pigs[Now].type << ":" << Pigs[Now].Pos << "使用无懈可击" << std::endl;
							#endif
							enemies(i, Now);
							return !query_wxkj(i, i, false);
						}
					}
				}
				i = nxt_pig(i);
			} while (i != start);
			return false;
		}
	}
	
	
	void jd(int From, int To, std::vector<Card>::iterator it) { //决斗 
		Pigs[From].cards.erase(it);
		#ifdef DEBUG
		std::cout << std::endl;
		std::cout << Pigs[From].type << ":" << Pigs[From].Pos << "对" << Pigs[To].type << ":" << Pigs[To].Pos << "使用决斗" << std::endl;
		#endif
		enemies(From, To, false);
		
		if (query_wxkj(From, To, true)) return ;
		int Now = From, Past = To;
		while (true) {
			std::swap(Now, Past);
			if (Pigs[Now].type == ZP && Pigs[Past].type == MP) break;
			if (!Pigs[Now].use_kill()) break;
		}
		#ifdef DEBUG
		std::cout << std::endl;
		std::cout << Pigs[Now].type << ":" << Pigs[Now].Pos << "没有出杀,收到伤害" << std::endl;
		#endif
		hurt(Past, Now, false, true);
	}
	
	void nzrq(int From) { //南猪入侵 
		#ifdef DEBUG
		std::cout << std::endl;
		std::cout << Pigs[From].type << ":" << Pigs[From].Pos << "使用南猪入侵" << std::endl;
		#endif
		for (int i = nxt_pig(From); i != From; i = nxt_pig(i)) {
			if (query_wxkj(From, i, true)) continue; 
			if (!Pigs[i].use_kill()) {
				#ifdef DEBUG
				std::cout << std::endl;
				std::cout << Pigs[i].type << ":" << Pigs[i].Pos << "没有出杀,收到伤害" << std::endl;
				#endif
				hurt(From, i, true);
			}
		} 
	} 
	
	void wjqf(int From) { //万箭齐发 
		#ifdef DEBUG
		std::cout << std::endl;
		std::cout << Pigs[From].type << ":" << Pigs[From].Pos << "使用万箭齐发" << std::endl;
		#endif
		for (int i = nxt_pig(From); i != From; i = nxt_pig(i)) {
			if (query_wxkj(From, i, true)) continue; 
			if (!Pigs[i].use_dodge()) {
				#ifdef DEBUG
				std::cout << std::endl;
				std::cout << Pigs[i].type << ":" << Pigs[i].Pos << "没有出闪,收到伤害" << std::endl;
				#endif
				hurt(From, i, true);
			}
		} 
	} 
}

namespace Turn {
	
	bool use_kill(int Now, std::vector<Card>::iterator it) { //尝试使用杀 
		int Nxt = nxt_pig(Now);
		#ifdef DEBUG
		std::cout << std::endl;
		std::cout << Pigs[Now].type << ":" << Pigs[Now].Pos << "尝试使用杀" << std::endl; 
		std::cout << "下一只猪:" << Pigs[Nxt].type << ":" << Pigs[Nxt].Pos << std::endl;
		#endif
		if (Now == Nxt) return false;
		if (Pigs[Now].type == MP) {
			if (Pigs[Nxt].Jump < UK) {
				Action::kill(Now, Nxt, it);
				return true;
			} else return false;
		}
		if (Pigs[Now].type == ZP) {
			if (Pigs[Nxt].Jump == TF) {
				Action::kill(Now, Nxt, it);
				return true;
			} else return false;
		}
		if (Pigs[Now].type == FP) {
			if (Pigs[Nxt].Jump > UK) {
				Action::kill(Now, Nxt, it);
				return true;
			} else return false;
		}
		return 0;
	}
	
	bool use_jd(int Now, std::vector<Card>::iterator it) { //尝试使用决斗
		#ifdef DEBUG
		std::cout << std::endl;
		std::cout << Pigs[Now].type << ":" << Pigs[Now].Pos << "尝试使用决斗" << std::endl; 
		#endif
		if (Pigs[Now].type == MP) {
			for (int i = nxt_pig(Now); i != Now; i = nxt_pig(i)) if (Pigs[i].Jump < UK) {
				Action::jd(Now, i, it);
				return true;
			} 
			else return false;
		}
		if (Pigs[Now].type == ZP) {
			for (int i = nxt_pig(Now); i != Now; i = nxt_pig(i)) {
				if (Pigs[i].Jump == TF) {
					Action::jd(Now, i, it);
					return true;
				}
			}
			return false;
		}
		if (Pigs[Now].type == FP) {
			Action::jd(Now, Main_Pig, it);
			return true;
		}
		return 0;
	}
	
	void turn(int Now) { //进行回合 
		for (int i = 0; i < 2; i++) Pigs[Now].insert(Pool.get_card());
		#ifdef DEBUG
		std::cout << std::endl;
		std::cout << "现在轮到" << Pigs[Now].type << ":" << Pigs[Now].Pos << " :" << std::endl;
		std::cout << "现在血量:" << Pigs[Now].HP << std::endl; 
		std::cout << "现在拥有的牌:" << Pigs[Now] << std::endl;
		#endif 
		bool Can_Use_Kill = true;
		
		bool used = true;
		while (used) {
			used = false;
			for (std::vector<Card>::iterator it = Pigs[Now].begin(); it != Pigs[Now].end(); it++) {
				if (it -> is_zgln()) { //猪哥连弩 
					Pigs[Now].insert_zgln(it);
					used = true;
					break;
				}
				if (it -> is_peach()) { //桃 
					if (Pigs[Now].use_peach()) {
						used = true;
						break;
					}
				}
				
				if (it -> is_kill()) { //杀 
					if ((Can_Use_Kill || Pigs[Now].AK_47)) {
						if (use_kill(Now, it)) {
							Can_Use_Kill = false;
							used = true;
							break;
						}
					}
				}
				
				if (it -> is_jd()) { //决斗 
					if (use_jd(Now, it)) {
						used = true;
						break;
					}
				}
				
				if (it -> is_nzrq()) { //南猪入侵 
					Pigs[Now].cards.erase(it);
					Action::nzrq(Now);
					used = true;
					break;
				}
				
				if (it -> is_wjqf()) { //万箭齐发 
					Pigs[Now].cards.erase(it);
					Action::wjqf(Now);
					used = true;
					break;
				}
			}
			
		}
	}
	
}


namespace Game {
	using std::cin;
	using std::cout;
	int Cards_Num;
	
	void start() {
		std::cin >> Pigs_Num >> Cards_Num;
		for (int i = 1; i <= Pigs_Num; i++) {
			string __type;
			cin >> __type;
			Pigs[i] = Pig(__type, i);
			for (int j = 0; j < 4; j++) {
				cin >> __type;
				Pigs[i].insert(__type);
			}
		}
		for (int i = 1; i <= Cards_Num; i++) {
			string __type;
			cin >> __type;
			Pool.insert(__type);
		}
		
		for (int i = Main_Pig; !game_over(); i = nxt_pig(i)) if (!Pigs[i].dead) Turn::turn(i);
		end_game();
	}
}
int main() {
	#ifdef DEBUG
	Debug::init();
	freopen("LG2482.log", "w", stdout); 
	#endif
	Game::start();
	return 0;
} 

  

posted @ 2018-10-26 16:32  Memory_of_winter  阅读(286)  评论(0编辑  收藏  举报