面向對象-紙牌遊戲實驗

 實驗項目名稱:紙牌遊戲 

實驗目的:

設計一個簡單的CardGames程序,運用面向對象封裝、繼承、抽象類、抽象方法、多態、動態綁定等概念。

實驗目標與要求:

參考windows的紙牌遊戲

使用語言:java

實驗內容:

單人紙牌遊戲,牌桌上有7個堆共28張牌,第一堆1張牌,第二堆2張,。。。第77張,每一堆的第一張牌朝上,其他朝下。牌桌上還有4suitpiles,一個deck card堆和一個discard card堆,佈局如下(參考windows的紙牌遊戲)

 

程序的總體設計:

利用CRC (Class,Responsibility,and Collaboration) 方法來設計,由於實驗是自己一人獨立完成,在設計時沒有多人蔘與crc設計,自己扮演了多個角色。

UML圖:

 

 

 

 

 

主要代碼及其說明:

1.     發配算法:

位於solitaire.game.Game類裏面


  1. static
  2.         //初始化紙牌 
  3.         allCard = new ArrayList<Card>(); 
  4.         for (int i = 0; i < 4; i++) 
  5.             for (int j = 0; j <= 12; j++)  
  6.                 allCard.add(new Card(j, i)); 
  7.         Random generator = new Random(); 
  8.         for (int i = 0; i < 52; i++) { 
  9.             int j = Math.abs(generator.nextInt() % 52); 
  10.                 // swap the two card values 
  11.             Card temp = allCard.get(i); 
  12.             allCard.set(i, allCard.get(j)); 
  13.             allCard.set(j, temp); 
  14.         } 
  15.         //初始化牌堆 
  16.         allPiles = new CardPile[13]; 
  17.         suitPile = new SuitPile[4]; 
  18.         tablePile = new TablePile[7]; 
  19.             // then fill them in 
  20.         allPiles[0] = deckPile = new DeckPile(20040); 
  21.         allPiles[1] = discardPile = new DiscardPile(200+Card.width+5040); 
  22.         for (int i = 0; i < 4; i++) 
  23.             allPiles[2+i] = suitPile[i] = 
  24.                 new SuitPile(200+Card.width+50 + Card.width + 150 + (40+Card.width) * i, 40); 
  25.         for (int i = 0; i < 7; i++) 
  26.             allPiles[6+i] = tablePile[i] = 
  27.                 new TablePile(200 + (50+Card.width) * i, 40 + Card.height + 40, i);  
  28.         for (int i = 0; i < 7; i++){ 
  29.             ArrayList<Card> al = new ArrayList<Card>(); 
  30.             for(int j = 0;j < tablePile[i].getCardNum();j++){ 
  31.                 al.add(allCard.remove(allCard.size()-1)); 
  32.             } 
  33.             tablePile[i].addCard(al); 
  34.             tablePile[i].setCardNum(tablePile[i].getNotFlipNum()+1); 
  35.             //System.out.println(tablePile[i].getCardNum()); 
  36.             tablePile[i].top().setFaceup(true); 
  37.         } 
  38.         int rest =  allCard.size(); 
  39.         for(int i = 0;i < rest;i++ ){ 
  40.             deckPile.addCard(allCard.remove(allCard.size()-1)); 
  41.             //System.out.println( i); 
  42.         } 
  43.         moveCard = new MoveCardPile(); 
  44.     } 

說明:該算法先創建52張紙牌(Card)對象,並放置allCardArrayList)中,然後模擬現實當中的洗牌操作,主要是利用javaRandom來打亂allCard裏牌的排列順序,然後初始化各個牌堆類: deckPile, discardPile, tablePile, suitPile, moveCard。此外還建了一個數組allPilesCardPile[]),用於存儲所有的牌堆類。最後將allCard中的紙牌牌(Card)對象分發至各個堆裏。

2. select方法(solitaire.pile. CardPile):

傳給該函數座標用於判斷點中該牌堆中的某張紙牌(還有個include方法用於判斷是否點中牌堆),其中因爲TablePile要支持選中多張紙牌,要改寫CardPile的方法。

CardPileselect方法:


  1. public int select (int tx, int ty) { 
  2.  
  3.     if(includes(tx,ty)){ 
  4.  
  5.        if(isEmpty()) 
  6.  
  7.            return -2
  8.  
  9.        else 
  10.  
  11.            return thePile.size() - 1
  12.  
  13.     } 
  14.  
  15.     else 
  16.  
  17.        return -1
  18.  

TablePileselect方法:


  1. public int select(int tx, int ty) { 
  2.  
  3.        // TODO Auto-generated method stub 
  4.  
  5.        if(!(isEmpty())){ 
  6.  
  7.        int beginX,beginY,endX,endY; 
  8.  
  9.        //System.out.println(notFlipNum+"  "+cardNum); 
  10.  
  11.        beginX = x  ; 
  12.  
  13.        beginY = y + unFlipCardSeparation * notFlipNum; 
  14.  
  15.        endX = x + Card.width; 
  16.  
  17.        endY =  beginY  + unFlipCardSeparation * notFlipNum + separation * (thePile.size() - 1 - notFlipNum) + Card.height; 
  18.  
  19.        boolean flip_include =  beginX <= tx && tx <= endX && beginY <= ty && ty <= endY; 
  20.  
  21.        //System.out.println(beginY+"   "+endY); 
  22.  
  23.        if(flip_include){ 
  24.  
  25.          
  26.  
  27.         int c = (ty - beginY)/separation + notFlipNum; 
  28.  
  29.         if(c >= thePile.size()){ 
  30.  
  31.            c =  thePile.size() - 1
  32.  
  33.         } 
  34.  
  35.         return c;//從零開始 
  36.  
  37.        } 
  38.  
  39.        else 
  40.  
  41.            return -1
  42.  
  43.        } 
  44.  
  45.        else 
  46.  
  47.            return -1
  48.  
  49.     } 
  50.  
  51.   

3. isCanAdd方法

用於判斷某張紙牌是否可以放於SuitePileTablePile

CardPile類:


  1. public boolean isCanAdd(Card card){ 
  2.  
  3.     return false
  4.  

SuitePile類:


  1. public boolean isCanAdd(Card card) { 
  2.  
  3.         // TODO Auto-generated method stub 
  4.  
  5.         if (isEmpty()) 
  6.  
  7.             return card.getNum() == 0; 
  8.  
  9.         Card toptopCard = top(); 
  10.  
  11.         return (card.getType() == topCard.getType()) && 
  12.  
  13.             (card.getNum() ==  topCard.getNum() + 1); 
  14.  
  15.         
  16.  
  17.     } 

Tablepile類:


  1. public boolean isCanAdd(Card card){ 
  2.  
  3.         // TODO Auto-generated method stub 
  4.  
  5.         if ( isEmpty()) 
  6.  
  7.             return card.getNum() == 12; 
  8.  
  9.         Card toptopCard = top(); 
  10.  
  11.         return (card.getColor() != topCard.getColor()) && 
  12.  
  13.             (card.getNum() ==  topCard.getNum()-1 ); 
  14.  
  15.     } 

4. refreshTablePile

用於刷新Tablepile,主要用於處理一走TablePile中的紙牌時剩餘紙牌都是背面的時候,將最上面的紙牌翻轉過來。


  1. public static void refreshTablePile(){ 
  2.  
  3.     //  System.out.println("refreshTablePile"); 
  4.  
  5.            
  6.  
  7.         for(int i=0;i<7;i++){ 
  8.  
  9.             if(tablePile[i].top() != null
  10.  
  11.             if(!(tablePile[i].top().isFaceup())){ 
  12.  
  13.                 tablePile[i].top().setFaceup(true); 
  14.  
  15.                 tablePile[i].setNotFlipNum(tablePile[i].getNotFlipNum()-1); 
  16.  
  17.             } 
  18.  
  19.         } 
  20.  
  21.     } 

實驗截圖:

 實驗心得:

結合課堂上的學習,利用了面向對象的思想:繼承,多態(改寫,多臺變量),範型等等,對面向對象思想有了更好的把握。

後來學習了設計原則,我覺得還得自己寫的代碼違背了許多原則,例如Game裏面的CardPile不是抽象的,應將其聲明爲抽象的以便讓其他類依賴抽象。此外我覺得可利用mvc架構來設計紙牌遊戲,利用觀察者模式來處理model的更改來更新view,用組成模式來設計view,利用策略模式來設計control。目前只初步設計了一下新的紙牌遊戲。

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章