Post

入侵式链表

简单看下跟常规链表不一样的入侵式链表,和相对而言,这种链表的优势

使用std::list<T>

看一个erase的场景

1
2
3
4
5
6
7
8
9
10
std::list<T> list;

void erase_node(auto* ptr) {
  for (auto it = list.begin(); it != list.end(); ++it) {
    if (&*it == ptr) {
      list.erase(it);
      break;
    }
  }
}

这种场景下,链表的删除显得非常的愚蠢,随机访问的效果很拉跨

使用入侵式链表

入侵链表要求link字段直接侵入被管理连接的对象中,侵入式链表让它存在于被链接的结构中。

比如

1
2
3
4
5
6
7
8
9
10
11
template <typename T>
struct IntrusiveListNode {
  IntrusiveListNode* prev;
  IntrusiveListNode* next;
  T* owner;
};

struct UserData {
  // ...
  InstruiveListNode list;
};

具体的对比可以看看boost的描述

优势

这里的优势一般都针对std::list<T*>来说的, 这种场景下入侵链表的优势会比较大

比如你要实现LRU cache,需要把entry同时加入HashMapList中,这时候如果用intrusive list的话,其实拿到entry就能操作链表了

  1. Data Locality,不用解引用(std::list<T*>),直接访问
  2. 脱离容器的生命周期,侵入式链表需要用户自己管理数据节点生命周期
  3. 对于侵入式链表,我们拿到数据就可以将这个节点从链表中摘除,而无需再去定位其 iterator,然后再去找到对应的容器 erase 这个 iterator。

实现

基本都是很简单的双向循环链表

linux里的实现

1
2
3
struct list_head {
  struct list_head *next, *prev;
};

使用方法见

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
// https://github.com/torvalds/linux/blob/v5.18/kernel/sched/sched.h#L376
struct task_group {
  // 省略一些业务逻辑
  struct list_head list;
};

// https://github.com/torvalds/linux/blob/v5.18/kernel/sched/core.c

/*
 * Default task group.
 * Every task in system belongs to this group at bootup.
 */
struct task_group root_task_group;
LIST_HEAD(task_groups);

list_add(&root_task_group.list, &task_groups);

c里数据结构比较简单,都能按照约定做layout,所以用几个宏就能拿到结构体的起始位置

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
// https://github.com/torvalds/linux/blob/v5.18/include/linux/list.h#L519

/**
 * list_entry - get the struct for this entry
 * @ptr:        the &struct list_head pointer.
 * @type:       the type of the struct this is embedded in.
 * @member:     the name of the list_head within the struct.
 */
#define list_entry(ptr, type, member) \
        container_of(ptr, type, member)

// https://github.com/torvalds/linux/blob/v5.18/include/linux/container_of.h#L17

/**
 * container_of - cast a member of a structure out to the containing structure
 * @ptr:          the pointer to the member.
 * @type:         the type of the container struct this is embedded in.
 * @member:       the name of the member within the struct.
 *
 */
#define container_of(ptr, type, member) ({                            \
        void *__mptr = (void *)(ptr);                                 \
        static_assert(__same_type(*(ptr), ((type *)0)->member) ||     \
                      __same_type(*(ptr), void),                      \
                      "pointer type mismatch in container_of()");     \
        ((type *)(__mptr - offsetof(type, member))); })

当然,c++就别用这个offset做事了,原因是c++的虚函数表,多继承等等,会让这个offset不对

c++里怎么办

看看doom3的实现

1
2
3
4
5
6
7
8
9
10
11
template< class type >
class idLinkList {
 public:
  // 省略

 private:
  idLinkList* head;
  idLinkList* next;
  idLinkList* prev;
  type*       owner;
};

用的时候类似这样

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
class idEntity {
  idLinkList<idEntity> spawnNode;  // for being linked into spawnedEntities list
};

idEntity::idEntity() {
  spawnNode.SetOwner( this );
}

/*
================
idLinkList<type>::InsertBefore

Places the node before the existing node in the list.  If the existing node is the head,
then the new node is placed at the end of the list.
================
*/
template< class type >
void idLinkList<type>::InsertBefore(idLinkList &node) {
  Remove();

  next       = &node;
  prev       = node.prev;
  node.prev  = this;
  prev->next = this;
  head       = node.head;
}

/*
================
idLinkList<type>::Remove

Removes node from list
================
*/
template< class type >
void idLinkList<type>::Remove( void ) {
  prev->next = next;
  next->prev = prev;

  next = this;
  prev = this;
  head = this;
}

/*
================
idLinkList<type>::AddToEnd

Adds node at the end of the list
================
*/
template< class type >
void idLinkList<type>::AddToEnd(idLinkList &node) {
  InsertBefore(*node.head);
}

ent->spawnNode.AddToEnd(spawnedEntities);

实际上,拿到owner就可以拿到具体的结构体了

REF

  1. Unrolled linked list
  2. C++ 侵入式链表总结
This post is licensed under CC BY 4.0 by the author.

Trending Tags