【蓝桥杯】 2017初赛 拉马车/STL队列,栈

题目描述

ps:题目有点冗长,简单来说就是,A,B用扑克牌玩接龙,输出最后赢家的牌,没有赢家输出-1。

小的时候,你玩过纸牌游戏吗?有一种叫做“拉马车”的游戏,规则很简单,却很吸引小朋友。
其规则简述如下:假设参加游戏的小朋友是A和B,游戏开始的时候,他们得到的随机的纸牌序列如下:
A方:[K, 8, X, K, A, 2, A, 9, 5, A]
B方:[2, 7, K, 5, J, 5, Q, 6, K, 4]
其中的X表示“10”,我们忽略了纸牌的花色。
从A方开始,A、B双方轮流出牌。
当轮到某一方出牌时,他从自己的纸牌队列的头部拿走一张,放到桌上,并且压在最上面一张纸牌上(如果有的话)。
此例中,游戏过程:A出K,B出2,A出8,B出7,A出X,此时桌上的序列为:K,2,8,7,X
当轮到B出牌时,他的牌K与桌上的纸牌序列中的K相同,则把包括K在内的以及两个K之间的纸牌都赢回来,放入自己牌的队尾。
注意:为了操作方便,放入牌的顺序是与桌上的顺序相反的。
此时,A、B双方的手里牌为:
A方:[K, A, 2, A, 9, 5, A]
B方:[5, J, 5, Q, 6, K, 4, K, X, 7, 8, 2, K]
赢牌的一方继续出牌。也就是B接着出5,A出K,B出J,A出A,B出5,又赢牌了。
5,K,J,A,5
此时双方手里牌:
A方:[2, A, 9, 5, A]
B方:[Q, 6, K, 4, K, X, 7, 8, 2, K, 5, A, J, K, 5]
注意:更多的时候赢牌的一方并不能把桌上的牌都赢走,而是拿走相同牌点及其中间的部分。
但无论如何,都是赢牌的一方继续出牌,有的时候刚一出牌又赢了,也是允许的。
当某一方出掉手里最后一张牌,但无法从桌面上赢取牌时,游戏立即结束。
对于本例的初始手牌情况下,最后A会输掉,而B最后的手里牌为:9K2A62KAX58K57KJ5
本题的任务就是已知双方初始牌序,计算游戏结束时,赢的一方手里的牌序。当游戏无法结束时,输出-1。

输入

输入存在多组数据,对于每组测试数据:
输入为2行,2个串,分别表示A、B双方初始手里的牌序列。输入的串的长度不超过30

输出

对于每组测试数据:输出为1行,1个串,表示A先出牌,最后赢的一方手里的牌序。

样例输入
96J5A898QA
6278A7Q973
25663K6X7448
J88A5KJXX45A

样例输出
2J9A7QA6Q6889977
6KAJ458KXAX885XJ645

【坑点总结】
1.模拟是肯定要模拟的,那么怎么判断是没有赢家的呢?
设置一个计数器去算AB出牌的次数,如果这个次数大于10000或者什么更大的数,差不多就无解了.一共才30张牌不至于这么久还没个胜负
2.用堆和栈来模拟这一过程,顺便来复习一下STL里面的队列栈函数:
函数详情

功能queuesatck
取第一个元素fronttop
弹出元素poppop
放入元素pushpush
是否为空emptyempty

3.我自己踩的一个坑,就是,我开始本来想如果vis那个函数为1的话,那就不把AB队列的元素弹出了吧,但是这就出大问题,因为从栈里面赢回来的元素是要放回队列尾部的,所以这样搞明显是错误的,不要想当然。
4.还是我自己踩的坑,就是那个vis函数,不仅是遇到相同的时候要把相同的弹出来,在从栈收牌会队列的时候也要逐一改vis 的值,因为vis的意思是在栈中是否有这个牌,所以你拿走的时候当然应该改掉vis了。仔细一点啊喂!
5.还是一个坑,就是这题它要多组输入,所以你在做一切是之前,要把所有的队列啊栈啊vis数组啊全部都清零一下,尤其是vis数组,因为会干扰到下一组数据模拟。顺便提一下,vis是map型的,用memest还不太行,要用xxx.clear。
6.再废话一句,这个题目不是很难,但是每个细节都要清清楚楚才好!好了,给自己加油!

代码

#include<bits/stdc++.h>
using namespace std;
typedef map<char,int> mp;
string ta;
string tb;

int main()
{
    queue<char> a;
    queue<char> b;
    stack<char> c;
    mp vis;
    int ans=0;
    while(cin>>ta>>tb)
    {
        while(!a.empty())
            a.pop();
        while(!b.empty())
            b.pop();
        while(!c.empty())
            c.pop();
        vis.clear();
        int la=ta.length();
        int lb=tb.length();
        for(int i=0;i<la;i++)
        {
            a.push(ta[i]);
        }
        for(int i=0;i<lb;i++)
        {
            b.push(tb[i]);
        }
        int tem=0;
        while(!a.empty()&&!b.empty())
        {

            char hh;
            if(tem==0)
            {
                hh=a.front();
                //cout<<"tem0---"<<hh<<endl;
                a.pop();
                c.push(hh);
                if(vis[hh]==1)
                {
                    char temch=c.top();
                    a.push(temch);
                    c.pop();
                      while(1)
                      {
                          temch=c.top();
                          a.push(temch);
                          c.pop();
                          vis[temch]=0;
                          if(temch==hh)
                            break;
                      }

                  vis[hh]=0;
                  tem=!tem;
                }
                else
                {
                    vis[hh]=1;
                }


            }
            else
            {

                hh=b.front();
                //cout<<"tem1---"<<hh<<endl;
                b.pop();
                c.push(hh);
                if(vis[hh]==1)
                {
                    char temch=c.top();
                    b.push(temch);
                    c.pop();
                      while(1)
                      {
                          temch=c.top();
                          b.push(temch);
                          c.pop();
                          vis[temch]=0;
                          if(temch==hh)
                            break;
                      }

                  vis[hh]=0;
                  tem=!tem;
                }
                else
                {
                    vis[hh]=1;
                }
            }

            tem=!tem;
            ans++;
            if(ans>10000)
            {
               cout<<-1<<endl;
               return 0;
            }
        }
        if(a.empty())
        {
            while(!b.empty())
            {
                  cout<<b.front();
                  b.pop();
            }
           cout<<endl;
        }
        else
        {
            while(!a.empty())
            {
                  cout<<a.front();
                  a.pop();
            }
           cout<<endl;
        }

    }

    return 0;
}

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值