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

隨筆 - 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>
            欧美一区二区福利在线| 欧美日韩午夜在线| 99re热这里只有精品免费视频| 免费在线欧美黄色| 亚洲成人自拍视频| 亚洲国产精品一区| 亚洲国产日韩欧美综合久久| 亚洲人体大胆视频| 亚洲一区在线观看视频 | 亚洲黄色免费| 一本色道精品久久一区二区三区 | 亚洲女性裸体视频| 亚洲一区二区精品在线| 亚洲精品视频啊美女在线直播| 亚洲乱码国产乱码精品精| 日韩亚洲国产欧美| 久久都是精品| 亚洲国产一区在线观看| 亚洲小视频在线观看| 久久麻豆一区二区| 欧美日韩一区二| 红桃视频欧美| 亚洲女人天堂成人av在线| 蜜桃久久av一区| 亚洲视频免费看| 免费久久99精品国产自在现线| 国产精品日韩欧美大师| 91久久极品少妇xxxxⅹ软件| 性欧美xxxx大乳国产app| 欧美国产日韩精品| 欧美一级久久久| 欧美日韩免费观看中文| 亚洲国产精品成人精品| 欧美中日韩免费视频| 一本色道**综合亚洲精品蜜桃冫 | 久久视频在线看| 国产精品多人| 亚洲最黄网站| 欧美高潮视频| 久久三级福利| 在线国产精品播放| 免费短视频成人日韩| 欧美一级免费视频| 国产欧美日韩视频一区二区三区| 中文欧美在线视频| 免费成人av| 久久久999精品| 国产资源精品在线观看| 欧美一级片一区| 亚洲一区二区三区视频| 国产精品毛片a∨一区二区三区| 一区二区三区国产精华| 亚洲精品一区二区三区婷婷月 | 激情综合在线| 久久激情一区| 欧美在线观看视频在线| 国产一区二区av| 久久久美女艺术照精彩视频福利播放| 亚洲一区自拍| 国产亚洲高清视频| 久久一区激情| 老色批av在线精品| 亚洲精品国产精品久久清纯直播 | 欧美三级午夜理伦三级中视频| 在线成人h网| 欧美成年人视频| 欧美高清视频一二三区| 日韩一区二区精品葵司在线| 亚洲娇小video精品| 欧美久久在线| 亚洲欧美国产三级| 亚洲欧美综合| 极品少妇一区二区三区精品视频 | 亚洲欧美日韩在线一区| 国产日韩欧美在线播放不卡| 久久精品天堂| 欧美顶级艳妇交换群宴| 在线一区二区三区四区五区| 亚洲婷婷国产精品电影人久久| 国产精品一区久久久| 美女国内精品自产拍在线播放| 免费亚洲一区| 欧美一级免费视频| 久久久久久久久久久成人| 亚洲乱码国产乱码精品精可以看| 日韩天堂av| 激情婷婷欧美| 99热在这里有精品免费| 国产视频欧美视频| 亚洲电影免费观看高清完整版在线| 欧美精品在线观看播放| 久久国产精品第一页| 欧美jjzz| 久久精品国产精品亚洲| 欧美刺激性大交免费视频| 午夜精品国产更新| 女同性一区二区三区人了人一| 亚洲永久字幕| 女主播福利一区| 欧美一区在线视频| 欧美乱大交xxxxx| 久久久久女教师免费一区| 欧美日本国产视频| 免费精品视频| 国产日韩久久| 一区二区三区日韩精品视频| 一区二区三区在线视频观看 | 亚洲欧美日韩精品久久亚洲区| 亚洲黄色一区二区三区| 亚洲天堂网站在线观看视频| 亚洲激情视频| 久久精品国产综合精品| 亚洲欧美一区二区三区在线| 免费在线成人| 蜜臀久久99精品久久久画质超高清| 国产精品久久久久久模特| 91久久在线观看| 1024日韩| 久久精品综合网| 久久av红桃一区二区小说| 欧美色精品天天在线观看视频| 欧美高清成人| 久久青草福利网站| 欧美韩国一区| 欧美国产在线观看| 国产揄拍国内精品对白| 日韩亚洲综合在线| 亚洲美女少妇无套啪啪呻吟| 久久精品72免费观看| 性色一区二区三区| 国产精品xvideos88| 亚洲乱码久久| 在线亚洲免费| 国产精品v片在线观看不卡| 亚洲美女福利视频网站| 一本久道久久综合婷婷鲸鱼| 欧美日本不卡| 一区二区三区www| 亚洲欧美日韩综合aⅴ视频| 国产精品激情| 午夜日韩av| 老鸭窝91久久精品色噜噜导演| 国产在线拍揄自揄视频不卡99 | 久久伊人免费视频| 欧美成人精品在线| 亚洲激情专区| 欧美日韩免费观看一区三区| 一区二区三区产品免费精品久久75 | 亚洲视频一二区| 欧美三级午夜理伦三级中视频| 一片黄亚洲嫩模| 欧美在线观看一区二区| 黄色成人片子| 欧美激情日韩| 亚洲自拍另类| 欧美国产精品人人做人人爱| 在线午夜精品自拍| 国产一在线精品一区在线观看| 久久综合国产精品| 亚洲最黄网站| 久久国产日本精品| 亚洲国产精品久久人人爱蜜臀| 欧美精品自拍| 亚洲一区三区电影在线观看| 久久久天天操| 99国内精品| 国产在线观看一区| 欧美黑人在线观看| 午夜国产精品影院在线观看| 噜噜噜91成人网| 一区二区三区日韩欧美精品| 国产欧美日韩免费| 欧美精品国产一区二区| 欧美一乱一性一交一视频| 欧美黄色免费| 久久国产精品一区二区三区| 亚洲精品视频中文字幕| 国产女人精品视频| 久久综合久久综合这里只有精品| 99一区二区| 欧美国产成人在线| 久久精品国产2020观看福利| 亚洲人体大胆视频| 国产亚洲欧美另类中文| 欧美日韩亚洲一区二区| 狼狼综合久久久久综合网| 亚洲一区二区三区在线视频 | 欧美在线免费视屏| 99视频精品在线| 免费观看日韩| 日韩视频在线观看免费| 久久全国免费视频| 亚洲免费一在线| 亚洲黑丝一区二区| 国语自产在线不卡| 国产亚洲综合性久久久影院| 国产精品久久福利| 国产精品久久波多野结衣| 欧美黑人一区二区三区| 蜜桃av噜噜一区|