青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

隨筆 - 74, 文章 - 0, 評論 - 26, 引用 - 0
數據加載中……

在 C/C++ 中如何構造通用的對象鏈表

一個簡化的問題示例

鏈表的難點在于必須復制鏈表處理函數來處理不同的對象,即便邏輯是完全相同的。例如兩個結構類似的鏈表:


struct Struct_Object_A
  {
    int a;
    int b;
    Struct_Object_A *next;
  
  } OBJECT_A;
  
  typedef struct Struct_Object_B
  {
    int a;
    int b;
    int c;
    Struct_Object_B *next;
  
  } OBJECT_B; 

上面定義的兩個結構只有很小的一點差別。OBJECT_B 和 OBJECT_A 之間只差一個整型變量。但是,在編譯器看來,它們仍然是非常不同的。必須為存儲在鏈表中的每個對象復制用來添加、刪除和搜索鏈表的函數。為了解決這個問題,可以使用具有全部三個變量的一個聯合或結構,其中整數 c 并不是在所有的情況下都要使用。這可能變得非常復雜,并會形成不良的編程風格。

C 代碼解決方案:虛擬鏈表

此問題更好的解決方案之一是虛擬鏈表。虛擬鏈表是只包含鏈表指針的鏈表。對象存儲在鏈表結構背后。這一點是這樣實現的,首先為鏈表節點分配內存,接著為對象分配內存,然后將這塊內存分配給鏈表節點指針,如下所示:

虛擬鏈表結構的一種實現

typedef struct liststruct
  {
    liststruct *next;
  
  } LIST, *pLIST;
  
  pLIST Head = NULL;
  
  pLIST AddToList( pLIST Head,
void * data, size_t datasize )
  {
  pLIST newlist=NULL;
  void *p;
  
    // 分配節點內存和數據內存
    newlist = (pLIST) malloc
( datasize + sizeof( LIST ) );
  
    // 為這塊數據緩沖區指定一個指針
    p = (void *)( newlist + 1 );
  
    // 復制數據
    memcpy( p, data, datasize );
  
    // 將這個節點指定給鏈表的表頭
    if( Head )
    {
    newlist->next = Head;
    }
    else
    newlist->next = NULL;
  
    Head = newlist;
  
    return Head;
  }  

鏈表節點現在建立在數據值副本的基本之上。這個版本能很好地處理標量值,但不能處理帶有用 malloc 或 new 分配的元素的對象。要處理這些對象,LIST 結構需要包含一個一般的解除函數指針,這個指針可用來在將節點從鏈表中刪除并解除它之前釋放內存(或者關閉文件,或者調用關閉方法)。

一個帶有解除函數的鏈表

typedef void (*ListNodeDestructor)( void * );
  
  typedef struct liststruct
  {
    ListNodeDestructor DestructFunc;
    liststruct *next;
  
  } LIST, *pLIST;
  
  pLIST AddToList( pLIST Head, void * data, 
size_t datasize,
  ListNodeDestructor Destructor )
  {
  pLIST newlist=NULL;
  void *p;
  
  
    // 分配節點內存和數據內存
    newlist = (pLIST) malloc
( datasize + sizeof( LIST ) );
  
    // 為這塊數據緩沖區指定一個指針
    p = (void *)( newlist + 1 );
  
    // 復制數據
    memcpy( p, data, datasize );
  
    newlist->DestructFunc = Destructor;
    
    // 將這個節點指定給鏈表的表頭
    if( Head )
    {
      newlist->next = Head;
    }
    else
      newlist->next = NULL;
  
    Head = newlist;
  
    return Head;
  }
  
  void DeleteList( pLIST Head )
  {
    pLIST Next;
    while( Head )
    {
      Next = Head->next;
      Head->DestructFunc( 
(void *) Head );
      free( Head );
      Head = Next;
    }
  }
  
  typedef struct ListDataStruct
  {
    LPSTR p;
  
  } LIST_DATA, *pLIST_DATA;
  
  void ListDataDestructor( void *p )
  {
    // 對節點指針進行類型轉換
    pLIST pl = (pLIST)p;
  
    // 對數據指針進行類型轉換
    pLIST_DATA pLD = (pLIST_DATA)
( pl + 1 );
  
    delete pLD->p;
  }
  pLIST Head = NULL;
  
  void TestList()
  {
    pLIST_DATA d = new LIST_DATA;
    d->p = new char[24];
    strcpy( d->p, "Hello" ); 
    Head = AddToList( Head, (void *) d,
sizeof( pLIST_DATA ),
    ListDataDestructor );
    // 該對象已被復制,現在刪除原來的對象
    delete d;
  
    d = new LIST_DATA;
    d->p = new char[24];
    strcpy( d->p, "World" ); 
    Head = AddToList( Head, (void *) d,
sizeof( pLIST_DATA ),
    ListDataDestructor );
    delete d;
  
    // 釋放鏈表
    DeleteList( Head );
  }   

在每個鏈表節點中包含同一個解除函數的同一個指針似乎是浪費內存空間。確實如此,但只有鏈表始終包含相同的對象才屬于這種情況。按這種方式編寫鏈表允許您將任何對象放在鏈表中的任何位置。大多數鏈表函數要求對象總是相同的類型或類。

虛擬鏈表則無此要求。它所需要的只是將對象彼此區分開的一種方法。要實現這一點,您既可以檢測解除函數指針的值,也可以在鏈表中所用的全部結構前添加一個類型值并對它進行檢測。

當然,如果要將鏈表編寫為一個 C++ 類,則對指向解除函數的指針的設置和存儲只能進行一次。

C++ 解決方案:類鏈表

本解決方案將 CList 類定義為從 LIST 結構導出的一個類,它通過存儲解除函數的單個值來處理單個存儲類型。請注意添加的 GetCurrentData() 函數,該函數完成從鏈表節點指針到數據偏移指針的數學轉換。一個虛擬鏈表對象

// 定義解除函數指針
  
  typedef void (*ListNodeDestructor)
( void * );
  
  // 未添加解除函數指針的鏈表
  
  typedef struct ndliststruct
  {
    ndliststruct *next;
  
  } ND_LIST, *pND_LIST;
  
  // 定義處理一種數據類型的鏈表類
  
  class CList : public ND_LIST
  {
  public:
    CList(ListNodeDestructor);
    ~CList();
    pND_LIST AddToList
( void * data, size_t datasize );
    void *GetCurrentData();
    void DeleteList( pND_LIST Head );
  
  
  private:
    pND_LIST m_HeadOfList;
    pND_LIST m_CurrentNode;
    ListNodeDestructor
m_DestructFunc;
  };
  
  // 用正確的起始值構造這個鏈表對象
  
  CList::CList(ListNodeDestructor Destructor)
    : m_HeadOfList(NULL), 
m_CurrentNode(NULL)
  {
    m_DestructFunc = Destructor;
  }
  
  // 在解除對象以后刪除鏈表
  
  CList::~CList()
  {
    DeleteList(m_HeadOfList);
  }
  
  // 向鏈表中添加一個新節點
  
  pND_LIST CList::AddToList
( void * data, size_t datasize )
  {
  pND_LIST newlist=NULL;
  void *p;
  
  
    // 分配節點內存和數據內存
    newlist = (pND_LIST) malloc
( datasize + sizeof( ND_LIST ) );
  
    // 為這塊數據緩沖區指定一個指針
    p = (void *)( newlist + 1 );
  
    // 復制數據
    memcpy( p, data, datasize );
  
    // 將這個節點指定給鏈表的表頭
    if( m_HeadOfList )
    {
      newlist->next = m_HeadOfList;
    }
    else
      newlist->next = NULL;
  
    m_HeadOfList = newlist;
  
    return m_HeadOfList;
  }
  
  // 將當前的節點數據作為 void 類型返回,
以便調用函數能夠將它轉換為任何類型
  
  void * CList::GetCurrentData()
  {
    return (void *)(m_CurrentNode+1);
  }
  
  // 刪除已分配的鏈表
  
  void CList::DeleteList( pND_LIST Head )
  {
    pND_LIST Next;
    while( Head )
    {
      Next = Head->next;
      m_DestructFunc( (void *) Head );
      free( Head );
      Head = Next;
    }
  }
  
  // 創建一個要在鏈表中創建和存儲的結構
  
  typedef struct ListDataStruct
  {
    LPSTR p;
  
  } LIST_DATA, *pND_LIST_DATA;
  
  // 定義標準解除函數
  
  void ClassListDataDestructor( void *p )
  {
    // 對節點指針進行類型轉換
    pND_LIST pl = (pND_LIST)p;
  
    // 對數據指針進行類型轉換
    pND_LIST_DATA pLD = (pND_LIST_DATA)
( pl + 1 );
  
    delete pLD->p;
  }
  
  // 測試上面的代碼
  
  void MyCListClassTest()
  {
    // 創建鏈表類
  
    CList* pA_List_of_Data =
new CList(ClassListDataDestructor);
  
    // 創建數據對象
    
    pND_LIST_DATA d = new LIST_DATA;
    d->p = new char[24];
    strcpy( d->p, "Hello" ); 
  
    // 創建指向鏈表頂部的局部指針
  
    pND_LIST Head = NULL;
  
    //向鏈表中添加一些數據
  
    Head = pA_List_of_Data->AddToList
( (void *) d, 
    sizeof( pND_LIST_DATA ) );
    // 該對象已被復制,現在刪除原來的對象
    delete d;
  
    // 確認它已被存儲
    char * p = ((pND_LIST_DATA) pA_List_of_Data->GetCurrentData())->p;
  
    d = new LIST_DATA;
    d->p = new char[24];
    strcpy( d->p, "World" ); 
    Head = pA_List_of_Data->AddToList
( (void *) d, sizeof( pND_LIST_DATA ) );
    // 該對象已被復制,現在刪除原來的對象
    delete d;
  
    // 確認它已被存儲
    p = ((pND_LIST_DATA) 
pA_List_of_Data->GetCurrentData())->p;
  
    // 刪除鏈表類,析構函數將刪除鏈表
    delete pA_List_of_Data;
  }
  

小結

從前面的討論來看,似乎僅編寫一個簡單的鏈表就要做大量的工作,但這只須進行一次。很容易將這段代碼擴充為一個處理排序、搜索以及各種其他任務的 C++ 類,并且這個類可以處理任何數據對象或類(在一個項目中,它處理大約二十個不同的對象)。您永遠不必重新編寫這段代碼。

posted on 2006-06-26 19:37 井泉 閱讀(272) 評論(0)  編輯 收藏 引用 所屬分類: 數據結構

青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            欧美亚洲一区二区在线| 久久精品系列| 国产精品一区二区三区观看| 免费在线播放第一区高清av| 玖玖精品视频| 亚洲第一视频网站| 亚洲欧洲日产国产网站| av成人激情| 亚洲欧美一区二区视频| 久久精品一级爱片| 欧美区视频在线观看| 欧美色精品在线视频| 国产日韩精品在线观看| 影音先锋亚洲电影| 在线视频一区观看| 久久久久久网址| 亚洲精品久久在线| 校园春色国产精品| 欧美精品福利| 国产一区深夜福利| 一区二区三区日韩在线观看| 久久久久久夜| 在线亚洲成人| 久久久久久久久久久久久9999| 欧美久久久久中文字幕| 国产视频一区在线| 一本一道久久综合狠狠老精东影业 | 亚洲少妇在线| 久久国产精品高清| 欧美精品xxxxbbbb| 国产在线视频欧美一区二区三区| 一二美女精品欧洲| 欧美成人免费在线观看| 亚洲欧美国产一区二区三区| 欧美激情精品| 亚洲高清免费| 久久久久综合网| 亚洲一区二区三区四区五区黄 | 99国产精品久久久久老师| 欧美一级片一区| 欧美午夜宅男影院| 一本色道**综合亚洲精品蜜桃冫| 久久裸体视频| 亚洲女人av| 欧美性大战xxxxx久久久| 亚洲欧洲免费视频| 美女任你摸久久| 久久av一区二区三区亚洲| 亚洲欧美色一区| 国产精品久久9| 一道本一区二区| 日韩视频在线观看一区二区| 欧美福利一区二区| 男人的天堂亚洲在线| 国内一区二区在线视频观看| 欧美亚洲网站| 亚洲影院在线| 国产精品免费网站| 亚洲欧美在线高清| 亚洲一区国产一区| 国产伦理一区| 久久久99免费视频| 久久国产66| 永久免费视频成人| 欧美大片在线观看| 欧美激情自拍| 亚洲午夜高清视频| 亚洲素人在线| 国产一区二区三区日韩| 久久亚洲不卡| 免费影视亚洲| 一区二区高清在线| 在线视频精品一区| 国产精品无人区| 久久人人爽国产| 欧美国产精品日韩| 亚洲欧美在线另类| 久久不射网站| 亚洲精品日韩在线观看| 一区二区三区国产精华| 国产亚洲一级高清| 亚洲高清中文字幕| 国产精品免费看片| 欧美成人精品一区二区| 欧美极品aⅴ影院| 亚洲欧洲av一区二区| 久久精品国产亚洲高清剧情介绍| 91久久精品日日躁夜夜躁欧美 | 亚洲精品欧美精品| 国产精品视频yy9099| 猫咪成人在线观看| 欧美日韩午夜| 久久噜噜噜精品国产亚洲综合| 欧美成人四级电影| 欧美一区影院| 欧美—级高清免费播放| 欧美伊人精品成人久久综合97| 久久性天堂网| 翔田千里一区二区| 欧美99在线视频观看| 亚洲欧美中文另类| 欧美成人免费小视频| 久久精品一区二区三区不卡牛牛| 男人的天堂亚洲| 久久精品色图| 国产精品视频不卡| 亚洲精品日韩欧美| 亚洲国产91色在线| 欧美亚洲三级| 亚洲自拍偷拍网址| 欧美精品v日韩精品v国产精品 | 亚洲综合久久久久| 亚洲乱码国产乱码精品精天堂| 国产精品久久激情| 一区二区不卡在线视频 午夜欧美不卡' | 欧美三区在线| 亚洲高清二区| 国内精品久久久久久久影视蜜臀| 亚洲伦理一区| 亚洲精品视频免费观看| 久久人人九九| 卡通动漫国产精品| 国内自拍一区| 欧美亚洲日本国产| 欧美在线播放| 国产欧美综合一区二区三区| 亚洲视频axxx| 午夜精品一区二区三区在线| 欧美激情网友自拍| 亚洲国产高清一区| 亚洲黄色av| 欧美成人一品| 亚洲激情女人| 一区二区不卡在线视频 午夜欧美不卡在 | 好看不卡的中文字幕| 欧美一级大片在线观看| 久久精品一二三| 国语自产精品视频在线看| 欧美在线观看你懂的| 久久国产精品黑丝| 国产日韩欧美综合| 欧美在线播放视频| 久久琪琪电影院| 亚洲黄色大片| 欧美日韩精品一区二区三区| 99视频有精品| 欧美一区视频在线| 激情欧美亚洲| 欧美激情自拍| 亚洲调教视频在线观看| 欧美在线综合| 亚洲激情在线观看视频免费| 欧美国产日韩视频| 一区二区三区视频观看| 欧美在线观看你懂的| 国产在线麻豆精品观看| 久久一区二区三区国产精品| 亚洲成色www8888| 亚洲婷婷免费| 国产自产高清不卡| 欧美岛国激情| 午夜欧美不卡精品aaaaa| 米奇777在线欧美播放| 亚洲另类黄色| 国产精品一区视频网站| 久久久噜噜噜久久久| 亚洲精品免费网站| 欧美一区在线视频| 亚洲看片网站| 国产最新精品精品你懂的| 欧美大片第1页| 亚洲欧美综合一区| 亚洲国产欧美日韩精品| 午夜久久影院| 亚洲精品资源| 亚洲第一精品福利| 国产在线观看精品一区二区三区 | 激情成人在线视频| 欧美日韩国产一区精品一区| 亚洲一区二区三区精品动漫| 欧美va天堂va视频va在线| 亚洲视频导航| 亚洲激情成人在线| 国产欧美精品久久| 欧美精品一区二区在线观看| 久久成人资源| 亚洲免费在线电影| 日韩视频精品在线观看| 蜜桃av一区二区| 香蕉久久夜色| 一区二区三区毛片| 亚洲国产精品嫩草影院| 国产日韩欧美黄色| 国产精品xnxxcom| 欧美精品aa| 欧美多人爱爱视频网站| 久久欧美中文字幕| 久久99在线观看| 欧美一级午夜免费电影| 亚洲一区视频|