美文网首页
一个用宏实现的List

一个用宏实现的List

作者: 明翼 | 来源:发表于2022-09-30 21:53 被阅读0次

一 C中宏的应用

如果我们写个排序方法, 用c语言实现的话,由于没有通用的类型,也没有c++的模板,没有java的泛化,这样不同的数据类型,我们就要实现多个排序方法,比如实现个int的排序方法,实现double类型的排序方法,但是这样重复的代码很多,其实可以用宏来实现。

同样有个需求要我们写个list,如果里面保存的数据类型不同,就要实现多个,这种实现方法会有太多代码重复,用宏实现是个不错的思路,用宏来实现相关函数还可以提升性能,
当然宏也有明显的缺点,比如不利于调试,写不好也容易在展开的时候出错。

二 内核中的一个list的实现

这个list 可以看作是双向链表,具体的实现如下:

#ifndef B4EDBD16_2948_4595_9135_6BBBAF9B6341
#define B4EDBD16_2948_4595_9135_6BBBAF9B6341
/*
 * Tail queue definitions.
 */
#define _TAILQ_HEAD(name, type, qual)                   \
struct name {                               \
    qual type *tqh_first;       /* first element */     \
    qual type *qual *tqh_last;  /* addr of last next element */ \
}
#define TAILQ_HEAD(name, type)  _TAILQ_HEAD(name, struct type,)

#define TAILQ_HEAD_INITIALIZER(head)                    \
    { NULL, &(head).tqh_first }

#define _TAILQ_ENTRY(type, qual)                    \
struct {                                \
    qual type *tqe_next;        /* next element */      \
    qual type *qual *tqe_prev;  /* address of previous next element */\
}
#define TAILQ_ENTRY(type)   _TAILQ_ENTRY(struct type,)
#define TAILQ_END(head)         NULL
#define TAILQ_FOREACH_SAFE(var, head, field, tvar)                  \
    for((var) = TAILQ_FIRST(head), \
        (tvar) = TAILQ_FIRST(head) ? TAILQ_NEXT(TAILQ_FIRST(head), field): NULL ; \
        (var) != TAILQ_END(head);                   \
        (var = tvar), (tvar) = var ? TAILQ_NEXT(var, field): NULL)
/*
 * Tail queue functions.
 */
#define TAILQ_INIT(head) do {                       \
    (head)->tqh_first = NULL;                   \
    (head)->tqh_last = &(head)->tqh_first;              \
} while (/*CONSTCOND*/0)

#define TAILQ_INSERT_HEAD(head, elm, field) do {            \
    if (((elm)->field.tqe_next = (head)->tqh_first) != NULL)    \
        (head)->tqh_first->field.tqe_prev =         \
            &(elm)->field.tqe_next;             \
    else                                \
        (head)->tqh_last = &(elm)->field.tqe_next;      \
    (head)->tqh_first = (elm);                  \
    (elm)->field.tqe_prev = &(head)->tqh_first;         \
} while (/*CONSTCOND*/0)

#define TAILQ_INSERT_TAIL(head, elm, field) do {            \
    (elm)->field.tqe_next = NULL;                   \
    (elm)->field.tqe_prev = (head)->tqh_last;           \
    *(head)->tqh_last = (elm);                  \
    (head)->tqh_last = &(elm)->field.tqe_next;          \
} while (/*CONSTCOND*/0)

#define TAILQ_INSERT_AFTER(head, listelm, elm, field) do {      \
    if (((elm)->field.tqe_next = (listelm)->field.tqe_next) != NULL)\
        (elm)->field.tqe_next->field.tqe_prev =         \
            &(elm)->field.tqe_next;             \
    else                                \
        (head)->tqh_last = &(elm)->field.tqe_next;      \
    (listelm)->field.tqe_next = (elm);              \
    (elm)->field.tqe_prev = &(listelm)->field.tqe_next;     \
} while (/*CONSTCOND*/0)

#define TAILQ_INSERT_BEFORE(listelm, elm, field) do {           \
    (elm)->field.tqe_prev = (listelm)->field.tqe_prev;      \
    (elm)->field.tqe_next = (listelm);              \
    *(listelm)->field.tqe_prev = (elm);             \
    (listelm)->field.tqe_prev = &(elm)->field.tqe_next;     \
} while (/*CONSTCOND*/0)

#define TAILQ_REMOVE(head, elm, field) do {             \
    if (((elm)->field.tqe_next) != NULL)                \
        (elm)->field.tqe_next->field.tqe_prev =         \
            (elm)->field.tqe_prev;              \
    else                                \
        (head)->tqh_last = (elm)->field.tqe_prev;       \
    *(elm)->field.tqe_prev = (elm)->field.tqe_next;         \
} while (/*CONSTCOND*/0)
#define TAILQ_REPLACE(head, elm, elm2, field) do {          \
    if (((elm2)->field.tqe_next = (elm)->field.tqe_next) != NULL)   \
        (elm2)->field.tqe_next->field.tqe_prev =        \
            &(elm2)->field.tqe_next;                \
    else                                \
        (head)->tqh_last = &(elm2)->field.tqe_next;     \
    (elm2)->field.tqe_prev = (elm)->field.tqe_prev;         \
    *(elm2)->field.tqe_prev = (elm2);               \
    _Q_INVALIDATE((elm)->field.tqe_prev);               \
    _Q_INVALIDATE((elm)->field.tqe_next);               \
} while (0)
#define TAILQ_FOREACH(var, head, field)                 \
    for ((var) = ((head)->tqh_first);               \
        (var);                          \
        (var) = ((var)->field.tqe_next))

#define TAILQ_FOREACH_REVERSE(var, head, headname, field)       \
    for ((var) = (*(((struct headname *)((head)->tqh_last))->tqh_last));    \
        (var);                          \
        (var) = (*(((struct headname *)((var)->field.tqe_prev))->tqh_last)))

#define TAILQ_CONCAT(head1, head2, field) do {              \
    if (!TAILQ_EMPTY(head2)) {                  \
        *(head1)->tqh_last = (head2)->tqh_first;        \
        (head2)->tqh_first->field.tqe_prev = (head1)->tqh_last; \
        (head1)->tqh_last = (head2)->tqh_last;          \
        TAILQ_INIT((head2));                    \
    }                               \
} while (/*CONSTCOND*/0)

/*
 * Tail queue access methods.
 */
#define TAILQ_EMPTY(head)       ((head)->tqh_first == NULL)
#define TAILQ_FIRST(head)       ((head)->tqh_first)
#define TAILQ_NEXT(elm, field)      ((elm)->field.tqe_next)

#define TAILQ_LAST(head, headname) \
    (*(((struct headname *)((head)->tqh_last))->tqh_last))
#define TAILQ_PREV(elm, headname, field) \
    (*(((struct headname *)((elm)->field.tqe_prev))->tqh_last))
#endif /* B4EDBD16_2948_4595_9135_6BBBAF9B6341 */

利用这个宏实现的小例子:

typedef struct _stu {
    char name[54];
    int age;
        // 在链表成员中的使用
    TAILQ_ENTRY(_stu) next;
} stu;

typedef struct _sch {
        // 链表
    TAILQ_HEAD(_stu_head, _stu) stu_list;
} sch;


int main(int argc, char **argv)
{
    sch *psch = (sch *)calloc(1, sizeof(sch));
         // 链表初始化
    TAILQ_INIT(& (psch->stu_list));

    stu s1 = {.name = "s1", .age = 1};
    stu s2 = {.name = "s2", .age = 2};
    stu s3 = {.name = "s3", .age = 3};
    stu s4 = {.name = "s4", .age = 4};
    stu s5 = {.name = "s5", .age = 5};
    stu s6 = {.name = "s6", .age = 6};

         // 在list前面插入
    TAILQ_INSERT_HEAD(&(psch->stu_list), &s1, next);
    TAILQ_INSERT_HEAD(&(psch->stu_list), &s2, next);

        // 在list的队尾插入
    TAILQ_INSERT_TAIL(&(psch->stu_list), &s3, next);
    TAILQ_INSERT_TAIL(&(psch->stu_list), &s5, next);

        // 在特定元素前面插入
    TAILQ_INSERT_BEFORE(&s5, &s4, next);
    TAILQ_INSERT_TAIL(&(psch->stu_list), &s6, next);
  printf("queue_head:%p, first:%p, last:%p\n", (void*)&(psch->stu_list), (void*)(psch->stu_list).tqh_first, (void*)(psch->stu_list).tqh_last);

    stu *p = NULL;
         // 正向遍历
    TAILQ_FOREACH(p, &(psch->stu_list), next) {
    printf("item:%p, next:%p, &next:%p, prev:%p, *prev:%p\n", p,p->next.tqe_next, &p->next.tqe_next,p->next.tqe_prev,& p->next.tqe_prev);
        printf("name:%s\t age:%d\n", p->name, p->age);
    }
    printf("---------------------------------\n");

       // 反向遍历
    TAILQ_FOREACH_REVERSE(p, &(psch->stu_list), _stu_head, next) {
        printf("name:%s\t age:%d\n", p->name, p->age);
    }   
        free(psch);
    return 0;
}

三 示意图说明

3.1 获取链表的最后一个元素

上述链表的示意图如下:


image.png

注意这个链表和我们一般自己在初学c语言的链表不一样,这个里面的tqe_prev指向的不是前一个元素,而是指向前一个元素的tqe_next, 注意这个tqe_next是在stu里面的next的成员里面的。

这个宏有个比较难以理解的地方,就是得到最后一个元素和得到前一个元素:

#define TAILQ_LAST(head, headname) \
    (*(((struct headname *)((head)->tqh_last))->tqh_last))
#define TAILQ_PREV(elm, headname, field) \
    (*(((struct headname *)((elm)->field.tqe_prev))->tqh_last))

我们拿一个来解释TAILQ_LAST,如下图展示:
假如有如下的list:


原图

获取最后一个元素的示意图如下:


TAILQ_LAST示意图

TAILQ_LAST分为两步:

1. 获取tqh_last 指针,即指到最后一个元素的tqe_next;
2. 通过最后一个元素的tqe_next 获取到指向最后一个元素即绿色的stu的地址。

第一步很容易理解,通过(head)->tqh_last 获取到最后一个元素的tqe_next.
第二步很难理解,因为它把这个地址强制转成队列,成员怎么转成队列那,那是因为两个都是指针,指针的大小都是一样的,而且都是两个指针变量,所以可以互转,转了之后可以看到如上图的对应关系,tqe_next 变成了队列的tqh_first 了、同样tqe_prev也就变成了tqh_last,而tqh_last 即tqe_prev指向黄色元素的tqe_next了,然后通过* 取值即变成了绿色的stu的地址,即最后一个元素的地址了。

3.2 插入元素

#define TAILQ_INSERT_TAIL(head, elm, field) do {            \
    (elm)->field.tqe_next = NULL;                   \
    (elm)->field.tqe_prev = (head)->tqh_last;           \
    *(head)->tqh_last = (elm);                  \
    (head)->tqh_last = &(elm)->field.tqe_next;          \
} while (/*CONSTCOND*/0)

再画个插入的示意图:


初始化队列

插入一个元素:


image.png

线条上的数字即插入的顺序。

再次插入一个元素:


插入第二个元素

最终形成的链表:


插入两个元素后

3.3 宏调试相关

宏在调试之前,我们可以先通过gcc命令进行展开,可以获取到展开宏的源码。

gcc -E -P main.c > main.expand.c

在gdb调试的时候,可以通过:

gdb) macro expand 宏

gcc编译的时候、添加-g3 便于调试宏

gcc -g3 

四 参考

https://www.361shipin.com/blog/1534986025145729024
https://www.getdami.com/questions/wei-dui-lie-dai-ma-zhong-de-tailq_lastzen-yao-li-jie-89811/

相关文章

  • 一个用宏实现的List

    一 C中宏的应用 如果我们写个排序方法, 用c语言实现的话,由于没有通用的类型,也没有c++的模板,没有java的...

  • container/list

    container/list list是一个双向链表。可以用来实现队列和栈结构。 用例

  • 又一个rust实现的双向list

    又一个rust实现的双向list 简要说明 在上一个list的实现中,重点是是用Option

  • 用一个宏实现求两个数中的最大数

    用一个宏实现求两个数中的最大数 最常见的实现方法   在面试或者笔试中,经常会碰到“用一个宏实现求两个数中的最大数...

  • golang list用法笔记

    依赖 遍历 go的list也是用双向循环链表实现的,在尾部追加用PushBack() 删除元素 删除使用list....

  • 图(数据结构), since 2020.02.26

    数据结构的实现 1 Edge list structure边列表 用非排序list分别保存一个图的顶点(V)和边(...

  • Linux 上的内置链表

    1 简介 在 Linux 的 中定义一系列操作不同链表的宏函数。如: 用来实现 list,tail queue、...

  • 2018-07-09顺序表实现栈

    栈的实现 ——直接用顺序表(列表list)进行 栈结构实现 栈可以用顺序表实现,也可以用链表实现。 栈的操作 St...

  • Java中List和ArrayList的区别

    List是一个接口,而ArrayList是List接口的一个实现类。 ArrayList类继承并实现了List接口...

  • python实现图的BFS遍历

    用adjacency list实现的BFS。 用2个list储存状态:先初始化color,把所有点设为0,即还没遍...

网友评论

      本文标题:一个用宏实现的List

      本文链接:https://www.haomeiwen.com/subject/splrartx.html