Post

构造一个c++双向链表数据结构

主要是针对old-new-thing-Constructing nodes of a hand-made linked list, how hard can it be?

最简单的就是

1
2
3
4
5
struct node
{
    node* prev;
    node* next;
};

更进一步的思考就是天然让他做一个环形链表

1
2
3
4
5
struct node
{
    node* prev = this;
    node* next = this;
};

如果还想继续搞一个构造,从一个已经存在的节点构造一个新的back insert节点,要注意的是,这里不会拿引用,或者const&, 这是为了避免错误的搞成了拷贝构造

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
struct node
{
    node* prev = this;
    node* next = this;

    node() = default;

    // Construct a node after a specific node
    node(node* other) :                      
        prev(other),                         
        next(other->next)                    
    {                                        
        prev->next = this;                   
        next->prev = this;                   
    }                                        
};

然后有了back insert,你也应该有before insert

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
struct node
{
    node* prev = this;
    node* next = this;

    node() = default;

    // Construct a node after a specific node
    node(node* other) :
        prev(other),
        next(other->next)
    {
        prev->next = this;
        next->prev = this;
    }

    // Construct a node before a specific node
    node(node* other) :                       
        prev(other->prev),                    
        next(other)                           
    {                                         
        prev->next = this;                    
        next->prev = this;                    
    }                                         
};

但这样就重名了,然后正常人的解决方案可能是factory func或者tag dispatch

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
struct node
{
    node* prev = this;
    node* next = this;

    node() = default;

    static node create_after(node* other) 
    {                                     
        node n{ other->prev, other };     
        n.prev->next = &n;                
        n.next->prev = &n;                
        return n;                         
    }                                     
                                          
    static node create_before(node* other)
    {                                     
        node n{ other, other->next };     
        n.prev->next = &n;                
        n.next->prev = &n;                
        return n;                         
    }                                     
};

// usage
node a;
node b = node::create_after(&a); // a->b
node c = node::create_before(&b); // a->c->b

nrvo在这里可能不work,盲猜

  1. 函数内部使用了返回对象的地址
  2. 要靠编译器

所以怎么说都不一定能很好的work,所以他又搞了tag dispatch的版本

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
struct place_after_t {};                       
inline constexpr place_after_t place_after{};  
                                               
struct place_before_t {};                      
inline constexpr place_before_t place_before{};

struct node
{
    node* prev = this;
    node* next = this;

    node() = default;

    // Construct a node after a specific node
    node(place_after_t, node* other) :
        prev(other),
        next(other->next)
    {
        prev->next = this;
        next->prev = this;
    }

    // Construct a node before a specific node
    node(place_before_t, node* other) :
        prev(other->prev),
        next(other)
    {
        prev->next = this;
        next->prev = this;
    }

    static node create_after(node* other) 
    {                                     
        return { place_after, other };    
    }                                     
                                          
    static node create_before(node* other)
    {                                     
        return { place_before, other };   
    }                                     
};

// Sample usage 1: Using tags
node a;
node b(place_after, &a); // list is a->b
node c(place_before, &b); // list is  a->c->b

// Sample usage 2: Using factories
node a;
node b = node::create_after(&a); // list is a->b
node c = node::create_before(&b); // list is  a->c->b

这个看起来也很怪,这玩意没道理是static的,node都有instance了才有理由去调用,不如直接搞成成员方法.

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
struct node
{
    node* prev = this;
    node* next = this;

    node() = default;

    node create_after() { return { place_after, this }; }
    node create_before() { return { place_before, this }; }

private:
    struct place_after_t {};
    inline constexpr place_after_t place_after{};

    struct place_before_t {};
    inline constexpr place_before_t place_before{};

    node(place_after_t, node* other) :
        prev(other),
        next(other->next)
    {
        prev->next = this;
        next->prev = this;
    }

    node(place_before_t, node* other) :
        prev(other->prev),
        next(other)
    {
        prev->next = this;
        next->prev = this;
    }
};

// Sample usage
node a;
node b = a.create_after(); // list is a->b
node c = b.create_before(); // list is a->c->b

实质上,整个走下来,还是需要点想法

This post is licensed under CC BY 4.0 by the author.

Trending Tags