2012-10-23 135 views
0

我正在生成描述兩個節點(稱爲type1和type2)之間拍賣場景的大型遊戲樹,該樹生成得非常好,直到我達到第四級節點爲止指向垃圾節點的指針

#include <iostream> 
    #include <vector> 
    #include <queue> 

    #define LEVEL 8 

    using namespace std; 

    class payoff_node{ 
     int agent_1; 
     int agent_2; 
    }; 

    class node{ 
     public: 
      /* constructor to set payoff_ptr to NULL */ 
      node() { 
       payoff_ptr = NULL; 
       parent = NULL; 
      } 

      /* print the information about the node */ 
      void printNode(){ 
       cout << "type: " << type << "\t" << "level: " << level << "\t" << "purse: $" << purse << "\t" << "parent_bid: " << parent_bid << "\t" << "parent:" << parent->getType() << "\t" << endl; 
       /*for (int i = 0; i < actions.size() ; i++){ 
        cout << actions[i] << endl; 
       }*/ 
      } 

      void setType(int t){ type = t; } 

      void setLevel(int l){ level = l; } 

      void setPurse(int p){ 
       purse = p; 

       /* Depending on the purse value the action space is decided */ 
       for(int i = 0; i <= purse ; i++){ 
        actions.push_back(new node); 
       } 
      } 

      void setParentBid(int bid){ parent_bid = bid; } 

      void setParent(node &parent_node){ parent = &parent_node; } 



      int getLevel(){ return level; } 

      int getParentBid(){ return parent_bid; } 

      int getPurse(){ return purse; } 

      node** getParent(){ return &parent; } 

      int getType(){ return type; } 


      vector<node*> actions; 



     private: 
      int type; 
      int level; 
      int purse; 

      int parent_bid; 
      node* parent; 

      payoff_node* payoff_ptr; 
    }; 

    int main(int argc, char* argv[]){ 
     bool D = false; 
     /* Make the root node and set its properties */ 
     node root_node; 
     root_node.setType(1); 
     root_node.setLevel(1); 
     root_node.setPurse(4); 
     root_node.setParentBid(0); 

    // root_node.printNode(); 
     queue<node> myQ; 
     myQ.push(root_node); 

     /* BFS like creation of perfect information tree */ 
     while(true){ 
      node &ext = myQ.front(); 
      if(D) 
       cout << "h1" << endl; 

      if(ext.getLevel() == LEVEL){ 
       break; 
      } 

      if(D) 
       cout << "h2" << endl; 

      int type = ext.getType(); 

      if(D) 
      cout << "h3" << endl; 
      for(int i = 0; i < ext.actions.size() ; i++){ 
       node &child = *(ext.actions[i]); 
      if(D) 
      cout << "h4" << endl; 
       //process(child); 

       /* set parent */ 
       child.setParent(ext); 
      if(D) 
      cout << "h5" << endl; 

       /* set type & items won so far */ 
       if((ext.getLevel()) % 2 == 0){ // i.e. even numbered level, then current round has ended 
        if(i > ext.getParentBid()){ 
         child.setType(type);  
        } 
        else{ 
         int type_val = (*(*(ext.getParent()))).getType() ; 
         child.setType(type_val); 
        } 
       } 
       else{ 
        if(type == 1){ 
         child.setType(2); 

        } 
        else{ 
         child.setType(1); 

        } 
       } 

      if(D) 
      cout << "h6" << endl; 
       /* set level */ 
       child.setLevel(ext.getLevel() + 1); 
      if(D) 
      cout << "h7" << endl; 
       /* set purse */ 
       if(child.getType() == ext.getType()){ 
        int val = ext.getPurse() - i; 
        if(val < 0){ 
         child.setPurse(0); 
        } 
        else{ 
         child.setPurse(val); 
        } 
       } 
       else{ 

        if(*(ext.getParent()) != NULL ){ 
         int val = (*(ext.getParent()))->getPurse() - ext.getParentBid(); 
         if(val < 0){ 
          child.setPurse(0); 
         } 
         else{ 
          child.setPurse(val); 
         } 
        } 
        else{ 
         child.setPurse(3); 
        } 
       } 
      if(D) 
      cout << "h8" << endl; 

       /* set parent bid */ 
       child.setParentBid(i); 


      if(D) 
      cout << "h9" << endl; 
       /* when the child is ready with all its attributes, add it to the queue */ 
       myQ.push(child); 
       child.printNode(); 
      } 

      cout << endl << endl << "----Next Set of Children----" << endl; 
      myQ.pop(); 

     } 



    return 0; 

    } 

程序掛起,在這條線child.setPurse(val);我相信通過下面的行

int val = (*(ext.getParent()))->getPurse() - ext.getParentBid(); 

是錯誤的。當*(ext.getParent())點一些垃圾節點。任何幫助將非常,因爲我沒有得到理解計算的值能夠計算出超過24小時現在。謝謝。

回答

1

隊列正在存儲node類型的對象。你正在使用指向隊列的指針。你不應該這樣做!當你從隊列中彈出時,你在每次通過你的主循環結束時做的事情就會毀掉你的對象。

看:

node &ext = myQ.front(); 

// etc... 

for(int i = 0; i < ext.actions.size() ; i++){ 
    node &child = *(ext.actions[i]); 
    child.setParent(ext); 

    // etc... 

    myQ.push(child); 
} 

myQ.pop(); // <-- POOF!! Every pointer to 'ext' is now invalid. 

每次添加到隊列中,將創建您的對象的副本。當您引用該副本時,您正在使用隊列的內部副本。最終該副本將不存在。當把這個指針設置爲每個孩子的父母時,你就會要求嚴重的麻煩。

這個不會更快爆炸的唯一原因是因爲您在node(清理actions)中沒有提供適當的析構函數而導致內存泄漏。如果你實現了一個,你還必須實現一個拷貝構造函數(或者通過創建一個私有的空拷貝構造函數來防止拷貝)。

你真正需要做的是改變你的隊列使用指針:

queue<node*> myQ; 
myQ.push(&root_node); 
// etc...