回 帖 发 新 帖 刷新版面

主题:扑克牌洗牌(大家都来交流吧)

大家好:
    我是一个刚学习C++不久的学生,在学习过程中碰到了一个有趣的问题,就是扑克牌的洗牌问题。这是课上的一个洗牌函数(DeckOfCards,shuffle,deal):
void shuffle()
{
     int row;//represents suit value of card
     int coloum;//represents face value of card
            
     //for each of the 52 cards, choose a slot of the deck randomly
     for( int card = 1; card <= 52; card++ ) 
     {
         do// choose a new random location until unoccupied slot is found
         {
             row = rand() % 4;//randomly select the row
             coloum = rand() % 13; //randomly select the coloum
         }while( deck[row][coloum] != 0 );//end do while//随机洗牌
                  
     //place card number in chosen slot of deck
     deck[row][coloum] = card;//洗出的牌用数组deck储存
     }//end for
}//end function shuffle
    以上是洗牌函数 shuffle :在数组deck[row][coloum]中填充数字 1 ,表示这张牌是洗完牌后将要发的第一张牌,这个过程一直持续下去,把数字2,3,...52随机填充倒数组deck中。但是,当数组deck开始用牌号进行填充时,一张牌可能被第二次选中( 即当deck[row][coloum]被选中时其值非0 )。如果洗过的牌被重复地随机选择,那么这种洗牌算法可能要执行无限长的时间。这就是所谓的无限延迟,也称饿死(starvation)
    为了排除无限延迟的可能性,我向各位征求一些更高效的洗牌算法。
    如果能分析出算法的优点与缺点,也欢迎赐教。

回复列表 (共2个回复)

沙发

上面的函数还有:
DeckOfCards(构造函数)用于初始化数组deck
       //DeckOfCards default constructor initializes deck
       DeckOfCards()
       {
            //loop through rows of deck
            for( int row = 0; row <= 3; row++ )
            {
                 //loop through columns of deck for current row
                 for( int column = 0; column <= 12; column++ )
                 {
                      deck[row][column] = 0;//initialize slot of deck to 0
                 }//end inner for
            }// end outer for
            
            srand( time( 0 ) );//seed random nmber generator
       }//end DeckOfCards default constructor
       
       // shuffle cards in deck
deal,用于定义牌及发牌
       //deal cards in deck
       void deal()
       {
            //initialize suit array
            static const char *suit[4] = { "Hearts", "Diamonds", "Clubs", "Spades" };
            
            //initialize face array
            static const char *face[13] = { "Ace", "Deuce", "Three", "Four", "Five", "Six",
                   "Seven", "Eight", "Nine", "Ten", "Jack", "Queen", "King" }; 
            //for each of the 52 cards 
            for( int card = 1; card <= 52; card++ )
            {
                 //loop through rows of deck
                 for( int row = 0; row <= 3; row++ )
                 {
                      //loop through coloums of deck for current row
                      for( int coloum = 0; coloum <= 12; coloum++ )
                      {
                           //if slot contains current card, display card
                           if( deck[row][coloum] == card )
                           {
                               cout << setw( 5 ) << right << face[coloum] << " of " 
                               << setw( 8 ) << left << suit[row] << ( card % 2 == 0 ? '\n' : '\t' );
                           }//end if
                      }//end innermost for
                 }//end inner for 
            }//end outer for
       }//end function deal
两个函数都封装在 类  DeckOfCards 中。

板凳

//这段代码将1--52这些数字存储在card数组中模拟扑克的52张牌,然后利用方法 fresh(int [] card)将数组中数据的顺序打乱后输出,模拟洗牌的过程。

class Demo
{
    public static void main(String[]args)
    {
        int [] card = new int[52];
        for(int i=0;i<card.length;i++)
            card[i] = i+1;
            
        Demo.print(card);
        Demo.fresh(card);
        Demo.print(card);            
    }
    
    public static void print(int [] card)
    {
        for(int i=0;i<card.length;i++)
            System.out.print (card[i]+"  ");
        
        System.out.println ();    
    }
    
    public static void fresh(int [] card)
    {
        for(int i=0;i<card.length*2;i++)
        {
            int index1 = (int)(Math.random()*card.length);
            int index2 = (int)(Math.random()*card.length);            
            int tmp = card[index1];card[index1]=card[index2];card[index2]=tmp;
        }
    }
}

我来回复

您尚未登录,请登录后再回复。点此登录或注册