list图解

struct list {
struct list *next, *prev;

};





static inline void
list_init(struct list *list)
{
list->next = list;
list->prev = list;
}




static inline int
list_empty(struct list *list)
{
return list->next == list;
}





static inline void
list_insert(struct list *link, struct list *new_link)
{
new_link->prev= link->prev;
new_link->next= link;
new_link->prev->next= new_link;
new_link->next->prev= new_link;
}


注意:该操作相对比较难理解。这里把link为list节点方便理解,实际上link是任意的,该操作是把新的节点插入到link节点的前方。

list初始为空时:此时2个节点前后都是一样的节点,也就是说无所谓前后了。



list初始不为空时:






static inline void
list_append(struct list *list, struct list *new_link)
{
list_insert((struct list *)list, new_link);
}

插入到当前节点的前方,也就是说让新的节点的prev指针指向当前节点的prev指针指向的地方,让新的节点的next指针指向当前节点。





static inline void
list_prepend(struct list *list, struct list *new_link)
{
list_insert(list->next, new_link);
}

插入到当前节点的前方,也就是说让新的节点的prev指针指向当前节点,让当前节点的next指针指向新的节点。




static inline void
list_remove(struct list *link)
{
link->prev->next = link->next;
link->next->prev = link->prev;
}









#define list_entry(link, type, member) \
((type *)((char *)(link)-(unsigned long)(&((type *)0)->member)))


在实际使用中,list往往是作为某个结构体的成员,通过list的地址,推导出结构体的地址。问题又回到通过结构体中的某个成员的地址推导出该结构体的地址的问题上来了。

type:结构体类型。

member:结构体的成员。

link:结构体成员member的地址。




#define list_head(list, type, member) \
list_entry((list)->next, type, member)

通过list节点的next指向的节点的地址,推导出list节点的next指向的节点所在的结构体的地址。




#define list_tail(list, type, member) \
list_entry((list)->prev, type, member)


通过list节点的prev指向的节点的地址,推导出list节点的prev指向的节点所在的结构体的地址。




#define list_next(elm, member) \
list_entry((elm)->member.next, typeof(*elm), member)

参考以上。



#define list_for_each_entry(pos, list, member) \
for (pos = list_head(list, typeof(*pos), member);\
    &pos->member != (list);\
    pos = list_next(pos, member))



遍历整个list双向链表,其中参数list作为一个哨兵节点的角色,以它来作为退出的条件。



边想边写,写得太乱了,头晕了。







  • 1
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Transformers是一种用于自然语言处理和其他相关领域的深度学习模型。它是通过编码器-解码器结构实现的。编码器将输入序列转换为一系列隐藏状态,而解码器则根据编码器的输出和之前的上下文生成输出序列。 关于图解transformer,引用中提到的图显示了Transformer的位置编码方法,其中将两个信号交织在一起。这个图可以帮助我们更好地理解位置编码的实现方式。 此外,引用中还展示了一个包含两个堆叠编码器和解码器的Transformer结构的图示。这个图可以帮助我们了解多层Transformer的组织结构。 最后,引用中提到的训练模型的直觉可以帮助我们更好地理解Transformer的工作原理。这个直觉的图示可能显示了一些与训练有关的信息,可以帮助我们更好地理解整个前向传递过程。 综上所述,通过引用中提到的图示,我们可以更好地理解Transformer模型的一些关键概念和操作。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* *2* *3* [【Transformer】图解 Transformer](https://blog.csdn.net/sikh_0529/article/details/128968765)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 100%"] [ .reference_list ]

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值