在前面的文章中,我们讨论了如何实现通用类型的链表,方法是用void *类型的指针,指向数据。那么还有其他的方法吗(不考虑内核链表)?

答案是肯定的。用零长数组也可以实现。

struct node_info
{
struct node_info *next;
struct node_info *prev;
char data[0];
};

这里的最后一个元素,是元素个数为0的数组。其不占用任何空间,甚至是一个指针的空间都不占!

注意:在标准C和C++中,长度为0的数组是被禁止使用的。不过在GNU C中,存在一个非常奇怪的用法,那就是长度为0的数组。

在一个结构体的最后 ,定义一个长度为0的数组,就可以使得这个结构体是可变长的。对于编译器来说,这个长度为0的数组并不占用空间,因为数组名本身不占空间,它只是一个偏移量, 数组名这个符号本身代
表了一个不可修改的地址常量

先来看看整个代码的头文件吧

#pragma once
struct node_info
{
struct node_info *next;
struct node_info *prev;
char data[0];
}; struct student
{
char name[20];
unsigned char age; };//for test //有头双向循环链表
struct dlist_info
{
struct node_info *head;
void (*add_head)(struct dlist_info *info,
const void *data, size_t size);
void (*add_tail)(struct dlist_info *info,
const void *data, size_t size);
void (*del)(struct node_info *node);
struct node_info* (*find)(struct dlist_info *info,
int (*compare)(void *dest_data, void *key_data), void *key_data);
void (*for_each_safe)(struct dlist_info *info,void (*todo)(struct node_info *)); }; int dlist_init(struct dlist_info *info);
void dlist_destroy(struct dlist_info *info); #define node_init(node) \
do\
{\
(node)->next = (node);\
(node)->prev = (node);\
}while(0) #define dlist_is_empty(info) \
((info)->head->next == (info)->head)

接下来我们实现一些方法

1.头插

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <assert.h>
#include "dlist.h" /* 有头循环双链表*/ static void dlist_add_head(struct dlist_info *info,
const void *my_data, size_t size)
{
assert(info != NULL && my_data != NULL); if (size == 0) {
return ;
} struct node_info *new_node = (struct node_info *)malloc(sizeof(struct node_info) + size); if (new_node == NULL) {
fprintf(stderr, "out of memory\n");
return ;
}
//数据域,内存拷贝
memmove(new_node->data, my_data, size); //指针域修改
new_node->next = info->head->next;
new_node->prev = info->head; info->head->next = new_node;
new_node->next->prev = new_node;
}

size 表示数据域占用了多少个字节。memmove(new_node->data, my_data, size); 这句话把用户的数据拷贝到了结构体的最后。关于指针域的修改,是不是有点绕呢?没有关系,画图就明白了。

2.尾插

static void dlist_add_tail(struct dlist_info *info,
const void *my_data, size_t size)
{ assert(info != NULL && my_data != NULL); if (size == 0) {
return ;
} struct node_info *new_node = (struct node_info *)malloc(sizeof(struct node_info) + size); if (new_node == NULL) {
fprintf(stderr, "out of memory\n");
return ;
}
//数据域,内存拷贝
memmove(new_node->data, my_data, size); //指针域修改
new_node->next = info->head;
new_node->prev = info->head->prev; info->head->prev->next = new_node;
info->head->prev = new_node; }

3.删除

static void dlist_del(struct node_info *node)
{
assert(node != NULL); node->next->prev = node->prev;
node->prev->next = node->next; node_init(node);
free(node);
}

因为申请空间的时候是带着size一起申请的,所以这里的释放就全部释放了,不存在内存泄漏。

4.查找

static struct node_info *dlist_find(struct dlist_info *info,
int (*key)(void *dest_data, void *key_data), void *key_data)
{
assert(info != NULL && key != NULL); if (dlist_is_empty(info)) {
fprintf(stderr, "dlist is empty\n");
return NULL;
} struct node_info *cur = NULL; for (cur = info->head->next; cur != info->head;
cur = cur->next) {
if (key(cur->data, key_data) != 0) {
return cur;
}
}
return NULL;
}

回调函数需要用户自己实现,不用多说。

5.安全遍历

static void dlist_for_each_safe(struct dlist_info *info,
void (*todo)(struct node_info *))
{
assert(info != NULL && todo != NULL); struct node_info *cur = NULL;
struct node_info *Next = NULL;
for (cur = info->head->next; cur != info->head;
cur = Next) {
Next = cur->next;
todo(cur);
}
}

6.构造和析构

int dlist_init(struct dlist_info *info)
{
info->head = (struct node_info *)malloc(sizeof(struct node_info)); if (info->head == NULL) {
fprintf(stderr, "Error:Out of memory\n");
return -1;
} /*头节点空间的初始化*/
node_init(info->head); /*函数指针的挂接*/
info->add_head = dlist_add_head;
info->add_tail = dlist_add_tail;
info->del = dlist_del;
info->find = dlist_find;
info->for_each_safe = dlist_for_each_safe;
return 0;
} void dlist_destroy(struct dlist_info *info)
{
// 依次删除,直到为空
while (!dlist_is_empty(info)) {
dlist_del(info->head->next);
} free(info->head);
}

接下来是单元测试。

测试一下头插和尾插吧。

#include "uni_test.h"

#include "dlist.h"

#include <stdio.h>

void setup (void)
{ // will excute in every case
} void teardown (void)
{ } void print_student(struct node_info *node)
{
struct student *p = (struct student *)(node->data);
printf("Name: %15s Age:%d\n",p->name,p->age); } int compare_student(void *dest,void *src)
{
struct student *p1 = dest;
struct student *p2 = src;
if(strcmp(p1->name,p2->name)==0)
return 1;
else
return 0;
} START_TEST(my_dlist_1)
{
struct student students[8] = {{"WangDong",18},{"LiuMing",19},{"SunYazhou",21},{"ChenYu",27},{"LiuXuewei",28},\
{"ZhangGuorong",47},{"LiuDehua",53},{"WangGuozhen",48}};
struct dlist_info list;
dlist_init(&list);
int i = 0;
for(;i<sizeof(students)/sizeof(students[0]);++i)
list.add_head(&list,students+i,sizeof(students[0]));
list.for_each_safe(&list,print_student);
printf("===========\n");
dlist_destroy(&list); }
END_TEST START_TEST(my_dlist_2)
{
struct student students[8] = {{"WangDong",18},{"LiuMing",19},{"SunYazhou",21},{"ChenYu",27},{"LiuXuewei",28},\
{"ZhangGuorong",47},{"LiuDehua",53},{"WangGuozhen",48}};
struct dlist_info list;
dlist_init(&list);
int i = 0;
for(;i<sizeof(students)/sizeof(students[0]);++i)
list.add_tail(&list,students+i,sizeof(students[0]));
list.for_each_safe(&list,print_student);
printf("===========\n");
dlist_destroy(&list); }
END_TEST

运行结果如图

Running suite(s): two_way_list_with_head

Name:     WangGuozhen  Age:48

Name:        LiuDehua  Age:53

Name:    ZhangGuorong  Age:47

Name:       LiuXuewei  Age:28

Name:          ChenYu  Age:27

Name:       SunYazhou  Age:21

Name:         LiuMing  Age:19

Name:        WangDong  Age:18

===========

Name:        WangDong  Age:18

Name:         LiuMing  Age:19

Name:       SunYazhou  Age:21

Name:          ChenYu  Age:27

Name:       LiuXuewei  Age:28

Name:    ZhangGuorong  Age:47

Name:        LiuDehua  Age:53

Name:     WangGuozhen  Age:48

===========


测试一下遍历,在遍历的过程中,我们把节点给删除了。这可以体现出安全遍历的好处。

START_TEST(my_dlist_3)//遍历删除
{
struct student students[8] = {{"WangDong",18},{"LiuMing",19},{"SunYazhou",21},{"ChenYu",27},{"LiuXuewei",28},\
{"ZhangGuorong",47},{"LiuDehua",53},{"WangGuozhen",48}};
struct dlist_info list;
dlist_init(&list);
int i = 0;
for(;i<sizeof(students)/sizeof(students[0]);++i)
list.add_tail(&list,students+i,sizeof(students[0]));
list.for_each_safe(&list,list.del); list.for_each_safe(&list,print_student);
printf("===========\n");
dlist_destroy(&list); }
END_TEST

运行结果是:

===========


果然没有节点了。

查找并删除。

START_TEST(my_dlist_4)//查找并删除
{
struct student students[8] = {{"WangDong",18},{"LiuMing",19},{"SunYazhou",21},{"ChenYu",27},{"LiuXuewei",28},\
{"ZhangGuorong",47},{"LiuDehua",53},{"WangGuozhen",48}};
struct dlist_info list;
dlist_init(&list);
int i = 0;
for(;i<sizeof(students)/sizeof(students[0]);++i)
list.add_tail(&list,students+i,sizeof(students[0])); list.del(list.find(&list,compare_student,"ChenYu")); list.for_each_safe(&list,print_student);
printf("===========\n"); dlist_destroy(&list); }
END_TEST

Name:        WangDong  Age:18

Name:         LiuMing  Age:19

Name:       SunYazhou  Age:21

Name:       LiuXuewei  Age:28

Name:    ZhangGuorong  Age:47

Name:        LiuDehua  Age:53

Name:     WangGuozhen  Age:48

===========


(完)

C语言实现通用链表初步(四)----双向链表的更多相关文章

  1. C语言实现通用链表初步(一)

    注意:本文讨论的是无头单向非循环链表. 假设不采用Linux内核链表的思路,怎样用C语言实现通用链表呢? 一种常用的做法是: typedef int element_t; struct node_in ...

  2. C语言实现通用链表初步(三)----单元测试

    前两节,我们已经完成了链表的一些操作,快来测试一下吧. 这里使用的单元测试工具名字叫"check". START_TEST(my_slist_1) { struct student ...

  3. C语言实现通用链表初步(二)

    接着上次的内容,我们继续! 还是无头单向非循环链表.假如要删除某个节点,如何实现? //删除成功返回0,失败返回-1 int slist_del(struct node_info *node, str ...

  4. (C语言版)链表(四)——实现双向循环链表创建、插入、删除、释放内存等简单操作

    双向循环链表是基于双向链表的基础上实现的,和双向链表的操作差不多,唯一的区别就是它是个循环的链表,通过每个节点的两个指针把它们扣在一起组成一个环状.所以呢,每个节点都有前驱节点和后继节点(包括头节点和 ...

  5. C语言数据结构-创建链表的四种方法

    结点类型: typedef int datatype; typedef struct NODE{ datatype data; struct NODE *next; }Node,*LinkList; ...

  6. 数组、单链表和双链表介绍 以及 双向链表的C/C++/Java实现

    概要 线性表是一种线性结构,它是具有相同类型的n(n≥0)个数据元素组成的有限序列.本章先介绍线性表的几个基本组成部分:数组.单向链表.双向链表:随后给出双向链表的C.C++和Java三种语言的实现. ...

  7. 拒绝造轮子!如何移植并使用Linux内核的通用链表(附完整代码实现)

    在实际的工作中,我们可能会经常使用链表结构来存储数据,特别是嵌入式开发,经常会使用linux内核最经典的双向链表 list_head.本篇文章详细介绍了Linux内核的通用链表是如何实现的,对于经常使 ...

  8. C语言实现单链表-03版

    在C语言实现单链表-02版中我们只是简单的更新一下链表的组织方式: 它没有更多的更新功能,因此我们这个版本将要完成如下功能: Problem 1,搜索相关节点: 2,前插节点: 3,后追加节点: 4, ...

  9. C语言实现单链表-02版

    我们在C语言实现单链表-01版中实现的链表非常简单: 但是它对于理解单链表是非常有帮助的,至少我就是这样认为的: 简单的不能再简单的东西没那么实用,所以我们接下来要大规模的修改啦: Problem 1 ...

随机推荐

  1. Vistual Studio的导出模板功能

    应用场景,每个项目有自己固定的目录结构和引用文件, 无需每次创建一个项目,就手工一一将那些目录再new一遍.如图所示 菜单  文件=>导出模板 之后的操作基本上一路"下一步" ...

  2. A - Chips

    Gerald plays the following game. He has a checkered field of size n × n cells, where m various cells ...

  3. IDEA 基本操作

    1.IDEA 编译的JDK问题 点击出错的模块 将这个改成8,但是改了这个还是不行,项目一编译他有回去了. 正确的做法: 在跟pom.xml 中增加如下代码: <build> <pl ...

  4. AngularJS(一)理论篇

    前言 大概今年春天的时候,自己对这个词产生了兴趣,那会只是简单的查了一下,并没有深入研究过这部分知识块,现在终于开始接触这些东西. 内容 AngularJS在web应用方面是一个非常完美的JavaSc ...

  5. 视图view没有主键,但可以添加唯一索引

    视图没有主键,但可以加上唯一索引 大致可以这样理解:视图是张虚拟的表.视图所对应的数据不进行实际的存储,数据库中只存储视图的定义,对视图的数据进行操作时,系统根据视图的定义去操作与视图相关联的基本表. ...

  6. 函数声明后面的const用法

    void function() const{} 通常我们会看到一些函数声明后面会跟着一个const,这个const是做什么的呢? 看一下下面的例子,就知道了.直接在编译前,就会提示下面的两个错误 // ...

  7. kuangbin专题十六 KMP&&扩展KMP HDU3068 最长回文

    给出一个只由小写英文字符a,b,c...y,z组成的字符串S,求S中最长回文串的长度. 回文就是正反读都是一样的字符串,如aba, abba等 Input输入有多组case,不超过120组,每组输入为 ...

  8. centos6.4安装过程中无法出现图形化界面

    今天在VMware9.0安装centos6.4的时候,前面的步骤都没问题,到跳过媒体检查这一步后,就发现和之前的不一样了,这次安装不是图形界面,导致后面的安装不方便 一开始以为是centos的版本不一 ...

  9. C语言关于++i,--i,i++,i--

    ++i 和--i 指的是先进行运算,再进行调用(运算符在前) i++和i--指的是先进行调用,再进行运算(运算符在后) 举例: int k,i=5;k=i++;//k得到5i=5;k=++i;//k得 ...

  10. JavaScript权威指南--立即执行函数

    千万不要停下追逐梦想的脚步 (function(){ //execute this method immediatly. //content... }());