通用双向链表的C语言实现

双向链表是一种基本的数据结构,在许多场合都会使用到。最近正在编写一个基于Linux的GUI系统,里边许多地方都要使用到双向链表,于是写了一个通用的双向链表。代码如下:

  1. #ifndef __STDLIST_H__
  2. #define __STDLIST_H__
  3. typedef struct tagSTDNODE       STDNODE, *LPSTDNODE;
  4. typedef struct tagSTDLIST       STDLIST, *LPSTDLIST;
  5. // 链表数据结构
  6. struct tagSTDNODE {
  7.     LPSTDNODE           lpPrev;
  8.     LPSTDNODE           lpNext;
  9. };
  10. // 链表节点结构
  11. struct tagSTDLIST {
  12.     LPSTDNODE           lpHead;
  13.     LPSTDNODE           lpTail;
  14. };
  15. // 用于链表扩展的宏
  16. #define NODE_INITIALIZER    ((STDNODE){ .lpPrev = NULL, .lpNext = NULL, })
  17. #define LIST_INITIALIZER    ((STDLIST){ .lpHead = NULL, .lpTail = NULL, })
  18. #define DECLARELISTNODE()  STDNODE __snStdNode
  19. #define INIT_LISTNODE()     __snStdNode = NODE_INITIALIZER
  20. // 初始化链表
  21. STATIC INLINE
  22. LPSTDLIST       TWINAPI     StdListInit(
  23.                             LPSTDLIST lpList
  24.                             ) {
  25.     lpList->lpHead  = NULL;
  26.     lpList->lpTail  = NULL;
  27.     return lpList;
  28. }
  29. // 获取链表头部节点
  30. STATIC INLINE
  31. LPSTDNODE       TWINAPI     StdListGetHeadNode(
  32.                             LPSTDLIST lpList
  33.                             ) {
  34.     return lpList->lpHead;
  35. }
  36. // 获取链表尾部节点
  37. STATIC INLINE
  38. LPSTDNODE       TWINAPI     StdListGetTailNode(
  39.                             LPSTDLIST lpList
  40.                             ) {
  41.     return lpList->lpTail;
  42. }
  43. // 获取给定节点的下一个节点
  44. STATIC INLINE
  45. LPSTDNODE       TWINAPI     StdListGetNextNode(
  46.                             LPSTDNODE lpNode
  47.                             ) {
  48.     return lpNode->lpNext;
  49. }
  50. // 获取给定节点的上一个节点
  51. STATIC INLINE
  52. LPSTDNODE       TWINAPI     StdListGetPrevNode(
  53.                             LPSTDNODE lpNode
  54.                             ) {
  55.     return lpNode->lpPrev;
  56. }
  57. // 在链表头部插入一个节点
  58. STATIC INLINE
  59. VOID            TWINAPI     StdListPushFront(
  60.                             LPSTDLIST lpList,
  61.                             LPSTDNODE lpNode
  62.                             ) {
  63.     lpNode->lpPrev  = NULL;
  64.     lpNode->lpNext  = lpList->lpHead;
  65.     if(lpList->lpHead)
  66.         lpList->lpHead->lpPrev  = lpNode;
  67.     else
  68.         lpList->lpTail  = lpNode;
  69.     lpList->lpHead  = lpNode;
  70. }
  71. // 在链表尾部插入一个节点
  72. STATIC INLINE
  73. VOID            TWINAPI     StdListPushBack(
  74.                             LPSTDLIST lpList,
  75.                             LPSTDNODE lpNode
  76.                             ) {
  77.     lpNode->lpNext  = NULL;
  78.     lpNode->lpPrev  = lpList->lpTail;
  79.     if(lpList->lpTail)
  80.         lpList->lpTail->lpNext  = lpNode;
  81.     else
  82.         lpList->lpHead  = lpNode;
  83.     lpList->lpTail  = lpNode;
  84. }
  85. // 在指定节点后插入一个节点
  86. STATIC INLINE
  87. VOID            TWINAPI     StdListInsert(
  88.                             LPSTDLIST lpList,
  89.                             LPSTDNODE lpAfter,
  90.                             LPSTDNODE lpNode
  91.                             ) {
  92.     if(lpAfter) {
  93.         if(lpAfter->lpNext)
  94.             lpAfter->lpNext->lpPrev = lpNode;
  95.         else
  96.             lpList->lpTail  = lpNode;
  97.         lpNode->lpPrev  = lpAfter;
  98.         lpNode->lpNext  = lpAfter->lpNext;
  99.         lpAfter->lpNext = lpNode;
  100.     } else {
  101.         StdListPushFront(lpList, lpNode);
  102.     }
  103. }
  104. // 从链表头部弹出一个节点
  105. STATIC INLINE
  106. LPSTDNODE       TWINAPI     StdListPopFront(
  107.                             LPSTDLIST lpList
  108.                             ) {
  109.     if(lpList->lpHead) {
  110.         LPSTDNODE lpNode    = lpList->lpHead;
  111.         if(lpList->lpHead->lpNext)
  112.             lpList->lpHead->lpNext->lpPrev  = NULL;
  113.         else
  114.             lpList->lpTail  = NULL;
  115.         lpList->lpHead  = lpList->lpHead->lpNext;
  116.         lpNode->lpPrev  = lpNode->lpNext    = NULL;
  117.         return lpNode;
  118.     } else {
  119.         return NULL;
  120.     }
  121. }
  122. // 从链表尾部弹出一个节点
  123. STATIC INLINE
  124. LPSTDNODE       TWINAPI     StdListPopBack(
  125.                             LPSTDLIST lpList
  126.                             ) {
  127.     if(lpList->lpTail) {
  128.         LPSTDNODE lpNode    = lpList->lpTail;
  129.         if(lpList->lpTail->lpPrev)
  130.             lpList->lpTail->lpPrev->lpNext  = NULL;
  131.         else
  132.             lpList->lpHead  = NULL;
  133.         lpList->lpTail  = lpList->lpTail->lpPrev;
  134.         lpNode->lpPrev  = lpNode->lpNext    = NULL;
  135.         return lpNode;
  136.     } else {
  137.         return NULL;
  138.     }
  139. }
  140. // 从链表中删除给定节点
  141. STATIC INLINE
  142. LPSTDNODE       TWINAPI     StdListRemove(
  143.                             LPSTDLIST lpList,
  144.                             LPSTDNODE lpNode
  145.                             ) {
  146.     if(lpNode->lpPrev)
  147.         lpNode->lpPrev->lpNext  = lpNode->lpNext;
  148.     else
  149.         lpList->lpHead  = lpNode->lpNext;
  150.     if(lpNode->lpNext)
  151.         lpNode->lpNext->lpPrev  = lpNode->lpPrev;
  152.     else
  153.         lpList->lpTail  = lpNode->lpPrev;
  154.     return lpNode;
  155. }
  156. // 检查链表是否为空
  157. STATIC INLINE
  158. BOOL            TWINAPI     StdListIsEmpty(
  159.                             LPSTDLIST lpList
  160.                             ) {
  161.     if(lpList->lpHead || lpList->lpTail)
  162.         return FALSE;
  163.     else
  164.         return TRUE;
  165. }
  166. // 获取链表中的节点数
  167. STATIC INLINE
  168. LONG            TWINAPI     StdListGetSize(
  169.                             LPSTDLIST lpList
  170.                             ) {
  171.     LONG lnSize = 0;
  172.     LPSTDNODE lpNode    = StdListGetHeadNode(lpList);
  173.     while(lpNode) {
  174.         ++ lnSize;
  175.         lpNode  = StdListGetNextNode(lpNode);
  176.     }
  177.     return lnSize;
  178. }
  179. // 合并两个链表
  180. STATIC INLINE
  181. LPSTDLIST       TWINAPI     StdListCombine(
  182.                             LPSTDLIST lpList1,
  183.                             LPSTDLIST lpList2
  184.                             ) {
  185.     if(!StdListIsEmpty(lpList2)) {
  186.         if(!StdListIsEmpty(lpList1)) {
  187.             lpList1->lpTail->lpNext = lpList2->lpHead;
  188.             lpList2->lpHead->lpPrev = lpList1->lpTail;
  189.             lpList1->lpTail = lpList2->lpTail;
  190.         } else {
  191.             lpList1->lpHead = lpList2->lpHead;
  192.             lpList1->lpTail = lpList2->lpTail;
  193.         }
  194.         lpList2->lpHead = lpList2->lpTail   = NULL;
  195.     }
  196.     return lpList1;
  197. }
  198. #endif//__STDLIST_H__

算法本身没有什么特别的地方,为什么说这是一个通用双向链表呢?奥妙就在那四个用于扩展的宏。利用这几个宏你可以将这个双向链表扩展到任意结构。

例如,我们要实现一个消息队列,那么我们可以这样定义消息节点:

  1. typedef struct tagMSGNODE {
  2.     DECLARELISTNODE();
  3.     HWND        hWnd;
  4.     UINT        uMsg;
  5.     WPARAM      wParam;
  6.     LPARAM      lParam;
  7. } MSGNODE, *LPMSGNODE;

而消息队列则可以直接用LISTNODE结构来表示:

  1. STDLIST     lstMsgQueue;

这样我们就可以用刚刚定义的函数来对这个消息队列进行操作了:

  1. // 向消息队列插入一个消息
  2. LPMSGNODE lpMsgNode = MemAlloc(SIZEOF(MSGNODE));
  3. lpMsgNode->INIT_LISTNODE();
  4. lpMsgNode->hWnd     = hWnd;
  5. lpMsgNode->uMsg     = WM_PAINT;
  6. lpMsgNode->wParam   = NULL;
  7. lpMsgNode->lParam   = NULL;
  8. StdListPushBack(&lstMsgQueue, (LPSTDNODE)lpMsgNode);
  9. // 从消息队列取出一个消息
  10. LPMSGNODE lpMsgNode = (LPMSGNODE)StdListPopFront(&lstMsgQueue);
  11. // 删除所有属于hWnd的消息
  12. LPMSGNODE lpTemp    = (LPMSGNODE)StdListGetHeadNode(&lstMsgQueue);
  13. while(lpTemp) {
  14.     LPMSGNODE lpNext    = (LPMSGNODE)StdListGetNextNode(&lstMsgQueue, (LPSTDNODE)lpTemp);
  15.     if(lpTemp->hWnd == hWnd) {
  16.         StdListRemove(&lstMsgQueue, (LPSTDNODE)lpTemp);
  17.                 MemFree(lpTemp);
  18.         }
  19.     lpTemp  = lpNext;
  20. }

是不是很方便呢^_^。

 

备注:

由于正在编写的GUI是参考Windows的体系结构,所以代码风格采用了微软的风格。

代码中的MemAlloc,MemFree函数以及各种数据结构均在其他地方定义。

 

 

评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值