BZOJ 1972: [SDOI2010]猪国杀
NanoApe
posted @ 2015年12月20日 02:23
in 蒟蒻整天被神题虐哭
, 1578 阅读
BZOJ恶心模拟题之一
做完感觉比杀蚂蚁要难得多(杀蚂蚁至少还有跟你说各个步骤的次序是怎么样的。。。
先给一份好看点的题目描述:
《猪国杀》是一种多猪牌类回合制游戏,一共有三种角色:主猪,忠猪,反猪。
每局游戏主猪有且只有一只,忠猪和反猪可以有多只,每只猪扮演一种角色。
游戏目的:
每局游戏主猪有且只有一只,忠猪和反猪可以有多只,每只猪扮演一种角色。
游戏目的:
主猪(MP):自己存活的情况下消灭所有的反猪。
忠猪(ZP):不惜一切保护主猪,胜利条件与主猪相同。
反猪(AP):杀死主猪。
游戏过程:
游戏开始时候,每个玩家手里都会有4张牌,且体力上限和初始体力都是4。
开始游戏时,从主猪开始,按照逆时针方向(数据中就是按照编号从1,2,3..n,1..的顺序)依次行动。
每个玩家自己的回合可以分为4个阶段:
◎摸牌阶段:从牌堆顶部摸两张牌,依次放到手牌的最右边;
◎出牌阶段:你可以使用0张到任意张牌,每次使用牌的时候都使用最靠左的能够使用的牌。当然,要满足如下规则:
◎摸牌阶段:从牌堆顶部摸两张牌,依次放到手牌的最右边;
◎出牌阶段:你可以使用0张到任意张牌,每次使用牌的时候都使用最靠左的能够使用的牌。当然,要满足如下规则:
- 如果没有猪哥连弩,每个出牌阶段只能使用一次“杀”来攻击;
- 任何牌被使用后被弃置(武器是装备上); 被弃置的牌以后都不能再用,即与游戏无关;
各种牌介绍:
每张手牌用一个字母表示,字母代表牌的种类。
◎基本牌:
『桃(P)』:
在自己的回合内,如果自己的体力值不等于体力上限,那么使用一个桃可以为自己补充一点体力,否则不能使用桃;
桃只能对自己使用;
在自己的回合外,如果自己的血变为0或者更低,那么也可以使用;
『杀(K)』:
在自己的回合内,对攻击范围内除自己以外的一名角色使用。
如果没有被『闪』抵消,则造成1点伤害。
无论有无武器,杀的攻击范围都是1;
『闪(D)』:
当你受到杀的攻击时,可以弃置一张闪来抵消杀的效果;
◎锦囊牌:
『决斗(F)』:
◎锦囊牌:
『决斗(F)』:
出牌阶段,对除自己以外任意一名角色使用,由目标角色先开始,自己和目标角色轮流弃置一张杀,首先没有杀可弃的一方受到1点伤害,另一方视为此伤害的来源;
『南猪入侵(N)』:
出牌阶段,对除你以外所有角色使用,按逆时针顺序从使用者下家开始依次结算,除非弃置一张杀,否则受到1点伤害;
『万箭齐发(W)』:
和南猪入侵类似,不过要弃置的不是杀而是闪;
『无懈可击(J)』:
在目标锦囊生效前抵消其效果。
每次有一张锦囊即将生效时,从使用这张锦囊的猪开始,按照逆时针顺序,依次得到使用无懈可击的机会;
效果:
用于决斗时,决斗无效并弃置;
用于南猪入侵或万箭齐发时,当结算到某个角色时才能使用,当前角色不需弃置牌并且不会受到伤害(仅对一个角色产生效果);
用于无懈可击时,成为目标的无懈可击被无效。
◎装备牌:
『猪哥连弩(Z)』:
◎装备牌:
『猪哥连弩(Z)』:
武器,出牌阶段你可以使用任意张杀;
同一时刻最多只能装一个武器;
如果先前已经有了一把武器,那么之后再装武器的话,会弃置以前的武器来装现在的武器;
特殊事件及概念解释:
特殊事件及概念解释:
◎伤害来源:
杀、南猪入侵、万箭齐发的伤害来源均是使用该牌的猪,决斗的伤害来源如上;
◎距离:
两只猪的距离定义为沿着逆时针方向间隔的猪数+1。
即初始时1和2的距离为1,但是2和1的距离就是n-1。
注意一个角色的死亡会导致一些猪距离的改变;
◎玩家死亡:
如果该玩家的体力降到0或者更低,并且自己手中没有足够的桃使得自己的体力值回到1,那么就死亡了,死亡后所有的牌(装备区,手牌区)被弃置;
◎奖励与惩罚:
反猪死亡时,最后一个伤害来源处(即使是反猪)立即摸三张牌。
忠猪死亡时,如果最后一个伤害来源是主猪,那么主猪所有装备牌、手牌被弃置;
◎注意,一旦达成胜利条件,游戏立刻结束,因此即使会摸3张牌或者还有牌可以用也不用执行了。
几种行为:
◎献殷勤:
使用无懈可击挡下南猪入侵、万箭齐发、决斗;
使用无懈可击抵消表敌意;
◎表敌意:
对某个角色使用杀、决斗;
使用无懈可击抵消献殷勤;
◎跳忠:
即通过行动表示自己是忠猪。
跳忠行动就是对主猪或对某只已经跳忠的猪献殷勤,或者对某只已经跳反的猪表敌意;
◎跳反:
即通过行动表示自己是反猪。
跳反行动就是对主猪或对某只已经跳忠的猪表敌意,或者对某只已经跳反的猪献殷勤;
忠猪不会跳反,反猪也不会跳忠;
不管是忠猪还是反猪,能够跳必然跳;
行动准则:
共性:
每个角色如果手里有桃且生命值未满,那么必然吃掉;
有南猪入侵、万箭齐发、必然使用;
有装备必然装上;
受到杀时,有闪必然弃置;
响应南猪入侵或者万箭齐发时候,有杀/闪必然弃置;
不会对未表明身份的猪献殷勤(包括自己);
特性:
◎主猪:
主猪会认为没有跳身份,且用南猪入侵/万箭齐发对自己造成伤害的猪是“类反猪”(没伤害到不算,注意“类反猪”并没有表明身份),如果之后跳了,那么主猪会重新认识这只猪;
对于每种表敌意的方式,对逆时针方向能够执行到的第一只“类反猪”或者已跳反猪表,如果没有,那么就不表敌意;
决斗时会不遗余力弃置杀;
如果能对已经跳忠的猪或自己献殷勤,那么一定献;
如果能够对已经跳反的猪表敌意,那么一定表;
◎忠猪:
对于每种表敌意的方式,对逆时针方向能够执行到的第一只已经跳反的猪表,如果没有,那么就不表敌意;
决斗时,如果对方是主猪,那么不会弃置杀,否则,会不遗余力弃置杀;
如果有机会对主猪或者已经跳忠的猪献殷勤,那么一定献;
◎反猪:
对于每种表敌意的方式,如果有机会则对主猪表,否则,对逆时针方向能够执行到的第一只已经跳忠的猪表,如果没有,那么就不表敌意;
决斗时会不遗余力弃置杀;
如果有机会对已经跳反的猪献殷勤,那么一定献。
行动准则Hzw的提醒(摘取):
1.此题注意样例少了个J,且牌堆可能用完牌,若牌用完则不停取最后一张
3.注意弃牌要弃装备
4.注意各种操作跳过已死的人
5.如果一个人打出无懈可击,则从他开始询问是否无懈无懈可击
8.对他人决斗可能导致自己死亡,要及时终止出牌阶段
嗯这题数据很大,不适宜跑数据检查错,对拍算是一种不错的选择
MyCode:
#include <cstdio>
#include <queue>
#define rep(i, l, r) for(int i=l; i<=r; i++)
#define down(i, l, r) for(int i=l; i>=r; i--)
#define clr(x, c) memset(x, c, sizeof(x))
#define maxn 19
#define maxm 209
using namespace std;
int P_Num, P_Fz, C_Rest, Id[maxn], IdNow[maxn], Hp[maxn], Nx[maxn], Pr[maxn], Now, WhoKill;
int Card[maxn][maxm], Sz[maxn];
bool Dead[maxn], Gameover, Zg[maxn];
queue<int>Rest; int RestEnd;
char ch; inline void Get(){ch=0; while (ch<'A' || ch>'Z') ch=getchar(); getchar();}
inline int GetId()
{
Get();
if (ch=='M') return 1;
if (ch=='Z') return 2;
if (ch=='F') return 3;
return -1;
}
inline int GetCard()
{
Get();
if (ch=='P') return 1;
if (ch=='K') return 2;
if (ch=='D') return 3;
if (ch=='F') return 4;
if (ch=='N') return 5;
if (ch=='W') return 6;
if (ch=='J') return 7;
if (ch=='Z') return 8;
return -1;
}
inline char Change(int x)
{
if (x==1) return 'P';
if (x==2) return 'K';
if (x==3) return 'D';
if (x==4) return 'F';
if (x==5) return 'N';
if (x==6) return 'W';
if (x==7) return 'J';
if (x==8) return 'Z';
return '9';
}
inline void PutoutCard(int x)
{
rep(i, 1, Sz[x]) printf("%d ", Card[x][i]);
puts("");
}
inline void DelCard(int x, int Use){Sz[x]--; rep(i,Use,Sz[x])Card[x][i]=Card[x][i+1];}
inline void AddCard(int x)
{
if (Rest.empty())
Card[x][++Sz[x]]=RestEnd;
else
Card[x][++Sz[x]]=Rest.front(), Rest.pop();
}
inline void Over(int x)
{
if (x==1) puts("FP"); else puts("MP");
rep(o, 1, P_Num)
{
if (Dead[o]) {puts("DEAD"); continue;}
rep(i, 1, Sz[o]) printf("%c%c", Change(Card[o][i]), (i==Sz[o])?'\n':' ');
if (Sz[o]==0) puts("");
}
Gameover=true;
}
inline bool Save(int x)
{
while (Hp[x]<=0)
{
bool Use=false;
rep(i, 1, Sz[x]) if (Card[x][i]==1) {Hp[x]++; DelCard(x,i); Use=true; break;}
if (!Use) break;
}
return Hp[x]>0;
}
inline void Die(int &x)
{
int tmp_x=Pr[x];
Dead[x]=1; if (Id[x]==3) P_Fz--;
Nx[Pr[x]]=Nx[x];
Pr[Nx[x]]=Pr[x];
if (Id[x]==1) Over(1);
if (P_Fz==0) Over(2);
if (Gameover) return;
if (Id[x]==3) {AddCard(WhoKill); AddCard(WhoKill); AddCard(WhoKill);}
if (Id[x]==2 && WhoKill==1) Sz[1]=0, Zg[1]=false;
x=tmp_x;
}
inline void ARedFlower(int x, int y)
{
if (Id[x]==2 && (Id[y]==1 || IdNow[y]==2)) IdNow[x]=2;
if (Id[x]==3 && IdNow[y]==3) IdNow[x]=3;
}
inline void DiuLeiLouMou(int x, int y)
{
if (Id[x]==3 && (Id[y]==1 || IdNow[y]==2)) IdNow[x]=3;
if (Id[x]==2 && IdNow[y]==3) IdNow[x]=2;
}
inline bool Will(int x, int y, int type)
{
if (type==0 && (Id[x]==1 || Id[x]==2) && (IdNow[y]==2 || IdNow[y]==1)) return 1;
if (type==0 && Id[x]==3 && IdNow[y]==3) return 1;
if (type==1 && (Id[x]==1 || Id[x]==2) && IdNow[y]==3) return 1;
if (type==1 && Id[x]==3 && (IdNow[y]==2 || IdNow[y]==1)) return 1;
return 0;
}
bool AskCancel(int x, int y, int type)
{
int tmp=x; while (true)
{
if (Will(tmp,y,type))
{
rep(i, 1, Sz[tmp]) if (Card[tmp][i]==7)
{
DelCard(tmp,i);
if (type==0) ARedFlower(tmp,y); else DiuLeiLouMou(tmp,y);
if (!AskCancel(tmp,y,type^1))
return true;
else
return false;
}
}
tmp=Nx[tmp]; if (tmp==x) break;
}
return false;
}
inline void Kill(int x, int y)
{
DiuLeiLouMou(x,y);
rep(i, 1, Sz[y]) if (Card[y][i]==3) {DelCard(y,i); return;}
Hp[y]--; WhoKill=x;
if (Hp[y]<=0 && !Save(y)) Die(y);
}
inline void Fight(int x, int y)
{
DiuLeiLouMou(x,y);
if (AskCancel(x,y,0)) return;
while (true)
{
if (Id[x]==1 && Id[y]==2) break;
bool Use=false;
rep(i, 1, Sz[y]) if (Card[y][i]==2) {DelCard(y,i); Use=true; break;}
if (Use==false) break;
swap(x,y);
}
Hp[y]--; WhoKill=x;
if (Hp[y]<=0 && !Save(y)) Die(y);
}
inline void KingShit(int x)
{
for(int tmp=Nx[x]; tmp!=x; tmp=Nx[tmp]) if (!AskCancel(x,tmp,0))
{
bool Use=false;
rep(i, 1, Sz[tmp]) if (Card[tmp][i]==2) {DelCard(tmp,i); Use=true; break;}
if (Use==true) continue;
if (Id[tmp]==1 && IdNow[x]==0) IdNow[x]=-1;
Hp[tmp]--; WhoKill=x;
if (Hp[tmp]<=0 && !Save(tmp)) Die(tmp);
if (Gameover) return;
}
}
inline void KingTreasure(int x)
{
for(int tmp=Nx[x]; tmp!=x; tmp=Nx[tmp]) if (!AskCancel(x,tmp,0))
{
bool Use=false;
rep(i, 1, Sz[tmp]) if (Card[tmp][i]==3) {DelCard(tmp,i); Use=true; break;}
if (Use==true) continue;
if (Id[tmp]==1 && IdNow[x]==0) IdNow[x]=-1;
Hp[tmp]--; WhoKill=x;
if (Hp[tmp]<=0 && !Save(tmp)) Die(tmp);
if (Gameover) return;
}
}
inline bool WillKill(int type, int x)
{
if (type==1) return IdNow[x]==-1 || IdNow[x]==3;
if (type==2) return IdNow[x]==3;
if (type==3) return IdNow[x]==1 || IdNow[x]==2;
return false;
}
int main()
{
scanf("%d%d", &P_Num, &C_Rest);
rep(i, 1, P_Num)
{
Id[i]=GetId(); Hp[i]=4; if (Id[i]==3) P_Fz++;
rep(j, 1, 4) Card[i][++Sz[i]]=GetCard();
}
rep(i, 1, C_Rest) Rest.push(RestEnd=GetCard());
IdNow[1]=1;
rep(i, 1, P_Num-1) Nx[i]=i+1; Nx[P_Num]=1;
rep(i, 2, P_Num) Pr[i]=i-1; Pr[1]=P_Num;
Now=1;
while (true)
{
AddCard(Now); AddCard(Now); int KillTime=0;
while (true)
{
//PutoutCard(Now);
bool Use=false; rep(i, 1, Sz[Now])
{
if (Card[Now][i]==1 && Hp[Now]<4)
{
DelCard(Now,i); Hp[Now]++;
Use=true; break;
}
if (Card[Now][i]==2 && WillKill(Id[Now],Nx[Now]) && (KillTime<1 || Zg[Now]))
{
DelCard(Now,i); Kill(Now,Nx[Now]); KillTime++;
if (Gameover) return 0;
Use=true; break;
}
if (Card[Now][i]==4)
{
int tmp=Nx[Now]; while (tmp!=Now && !WillKill(Id[Now],tmp)) tmp=Nx[tmp]; if (tmp==Now) continue;
if (Id[Now]==3) tmp=1;
DelCard(Now,i); Fight(Now,tmp);
if (Gameover) return 0;
if (Hp[Now]<=0) break;
Use=true; break;
}
if (Card[Now][i]==5)
{
DelCard(Now,i); KingShit(Now);
if (Gameover) return 0;
Use=true; break;
}
if (Card[Now][i]==6)
{
DelCard(Now,i); KingTreasure(Now);
if (Gameover) return 0;
Use=true; break;
}
if (Card[Now][i]==8)
{
DelCard(Now,i); Zg[Now]=true;
Use=true; break;
}
}
if (Use==0) break;
}
//弃牌阶段
Now=Nx[Now];
}
return 0;
}
这题能1A的人简直是神。。。
做这道题我被vector的erase坑,被题目各种坑点坑,前前后后改了得有10次吧QwQ
妈蛋
评论 (0)