.NET中自带的链表是LinkedList类,并且已经直接实现成了双向循环链表。

其节点类LinkedListNode的数据结构如下,数据项包括指示到某个链表的引用,以及左,右节点和值。

  1. public sealed class LinkedListNode<T>
  2. {
  3. internal LinkedList<T> list;
  4. internal LinkedListNode<T> next;
  5. internal LinkedListNode<T> prev;
  6. internal T item;
  7. }

另外,获取前一个节点和后一个节点的实现如下:注意:里面的if-else结构的意义是当前一个(后一个)节点不为空且不是头节点时才不返回null,这样做的意义是当链表内只有1个节点时,其prev和next是指向自身的。

  1. [__DynamicallyInvokable]
  2. public LinkedListNode<T> Next
  3. {
  4. [__DynamicallyInvokable] get
  5. {
  6. if (this.next != null && this.next != this.list.head)
  7. return this.next;
  8. else
  9. return (LinkedListNode<T>) null;
  10. }
  11. }
  12. [__DynamicallyInvokable]
  13. public LinkedListNode<T> Previous
  14. {
  15. [__DynamicallyInvokable] get
  16. {
  17. if (this.prev != null && this != this.list.head)
  18. return this.prev;
  19. else
  20. return (LinkedListNode<T>) null;
  21. }
  22. }

过有一个把链表置为无效的方法定义如下:

  1. internal void Invalidate()
  2. {
  3. this.list = (LinkedList<T>) null;
  4. this.next = (LinkedListNode<T>) null;
  5. this.prev = (LinkedListNode<T>) null;
  6. }

而LinkedList的定义如下:主要的两个数据是头节点head以及长度count。

  1. public class LinkedList<T> : ICollection<T>, IEnumerable<T>, ICollection, IEnumerable, ISerializable, IDeserializationCallback
  2. {
  3. internal LinkedListNode<T> head;
  4. internal int count;
  5. internal int version;
  6. private object _syncRoot;
  7. private SerializationInfo siInfo;
  8. private const string VersionName = "Version";
  9. private const string CountName = "Count";
  10. private const string ValuesName = "Data";

而对此链表的主要操作,包括:

  • 插入节点到最后,Add(),也是AddLast()。
  • 在某个节点后插入,AddAfter(Node, T)。
  • 在某个节点前插入,AddBefore(Node, T)。
  • 插入到头节点之前,AddFirst(T)。
  • 清除所有节点,Clear()。
  • 是否包含某个值,Contains(T),也就是Find()。
  • 查找某个节点的引用,Find()和FindLast()。
  • 复制到数组,CopyTo(Array)
  • 删除某个节点,Remove(T)。
另外有几个内部方法,用来支撑复用插入和删除操作:
  • 内部插入节点,InternalInsertNodeBefore()
  • 内部插入节点到空链表,InternalInsertNodeToEmptyList()
  • 内部删除节点,InternalRemoveNode()
  • 验证新节点是否有效,ValidateNewNode()
  • 验证节点是否有效,ValidateNode()
 
插入新节点到链表最后的代码如下:
  1. public void AddLast(LinkedListNode<T> node)
  2. {
  3. this.ValidateNewNode(node);
  4. if (this.head == null)
  5. this.InternalInsertNodeToEmptyList(node);
  6. else
  7. this.InternalInsertNodeBefore(this.head, node);
  8. node.list = this;
  9. }

插入操作的第一步是验证节点是否有效,即节点不为null,且节点不属于其他链表。

  1. internal void ValidateNewNode(LinkedListNode<T> node)
  2. {
  3. if (node == null)
  4. throw new ArgumentNullException("node");
  5. if (node.list != null)
  6. throw new InvalidOperationException(SR.GetString("LinkedListNodeIsAttached"));
  7. }

如果头节点为空,则执行插入到空链表的操作:将节点的next和prev都指向为自己,并作为头节点。

  1. private void InternalInsertNodeToEmptyList(LinkedListNode<T> newNode)
  2. {
  3. newNode.next = newNode;
  4. newNode.prev = newNode;
  5. this.head = newNode;
  6. ++this.version;
  7. ++this.count;
  8. }

如果头节点不为空,则执行插入到头节点之前(注:因为是双向链表,所以插到头节点之前就相当于插到链表的最后了),具体的指针指向操作如下:

  1. private void InternalInsertNodeBefore(LinkedListNode<T> node, LinkedListNode<T> newNode)
  2. {
  3. newNode.next = node;
  4. newNode.prev = node.prev;
  5. node.prev.next = newNode;
  6. node.prev = newNode;
  7. ++this.version;
  8. ++this.count;
  9. }

而插入新节点到指定节点之后的操作如下:同样还是调用的内部函数,把新节点插入到指定节点的下一个节点的之前。有点绕,但确实让这个内部函数起到多个作用了。

  1. public void AddAfter(LinkedListNode<T> node, LinkedListNode<T> newNode)
  2. {
  3. this.ValidateNode(node);
  4. this.ValidateNewNode(newNode);
  5. this.InternalInsertNodeBefore(node.next, newNode);
  6. newNode.list = this;
  7. }

而插入新节点到指定节点之前的操作如下:直接调用插入新节点的内部函数,另外还要判断指定的节点是否是头节点,如果是的话,要把头节点变成新的节点。

  1. public void AddBefore(LinkedListNode<T> node, LinkedListNode<T> newNode)
  2. {
  3. this.ValidateNode(node);
  4. this.ValidateNewNode(newNode);
  5. this.InternalInsertNodeBefore(node, newNode);
  6. newNode.list = this;
  7. if (node != this.head)
  8. return;
  9. this.head = newNode;
  10. }

把新链表插入到第一个节点(也就是变成头节点)的操作如下:如果链表为空就直接变成头节点,否则就插入到头节点之前,取代头节点。

  1. public void AddFirst(LinkedListNode<T> node)
  2. {
  3. this.ValidateNewNode(node);
  4. if (this.head == null)
  5. {
  6. this.InternalInsertNodeToEmptyList(node);
  7. }
  8. else
  9. {
  10. this.InternalInsertNodeBefore(this.head, node);
  11. this.head = node;
  12. }
  13. node.list = this;
  14. }

查找链表中某个值的操作如下:注意直接返回null的条件是头节点为空。然后就是遍历了,因为是双向链表,所以要避免死循环(遍历到头节点时跳出)。

  1. public LinkedListNode<T> Find(T value)
  2. {
  3. LinkedListNode<T> linkedListNode = this.head;
  4. EqualityComparer<T> @default = EqualityComparer<T>.Default;
  5. if (linkedListNode != null)
  6. {
  7. if ((object) value != null)
  8. {
  9. while (!@default.Equals(linkedListNode.item, value))
  10. {
  11. linkedListNode = linkedListNode.next;
  12. if (linkedListNode == this.head)
  13. goto label_8;
  14. }
  15. return linkedListNode;
  16. }
  17. else
  18. {
  19. while ((object) linkedListNode.item != null)
  20. {
  21. linkedListNode = linkedListNode.next;
  22. if (linkedListNode == this.head)
  23. goto label_8;
  24. }
  25. return linkedListNode;
  26. }
  27. }
  28. label_8:
  29. return (LinkedListNode<T>) null;
  30. }

删除某个节点的操作如下:

  1. public void Remove(LinkedListNode<T> node)
  2. {
  3. this.ValidateNode(node);
  4. this.InternalRemoveNode(node);
  5. }

同样,内部删除节点的实现如下:如果节点指向自己,说明是头节点,所以直接把头节点置null。然后就是指针的指向操作了。

  1. internal void InternalRemoveNode(LinkedListNode<T> node)
  2. {
  3. if (node.next == node)
  4. {
  5. this.head = (LinkedListNode<T>) null;
  6. }
  7. else
  8. {
  9. node.next.prev = node.prev;
  10. node.prev.next = node.next;
  11. if (this.head == node)
  12. this.head = node.next;
  13. }
  14. node.Invalidate();
  15. --this.count;
  16. ++this.version;
  17. }

而清空链表的操作如下:遍历链表,逐个设置为无效,最后将内部的头节点也置为null。

  1. public void Clear()
  2. {
  3. LinkedListNode<T> linkedListNode1 = this.head;
  4. while (linkedListNode1 != null)
  5. {
  6. LinkedListNode<T> linkedListNode2 = linkedListNode1;
  7. linkedListNode1 = linkedListNode1.Next;
  8. linkedListNode2.Invalidate();
  9. }
  10. this.head = (LinkedListNode<T>) null;
  11. this.count = 0;
  12. ++this.version;
  13. }
 
链表转数组的实现如下:首先判断入参的有效性,然后从头节点开始遍历,依次复制到数组中,直到头结点(尽头)。
  1. public void CopyTo(T[] array, int index)
  2. {
  3. if (array == null)
  4. throw new ArgumentNullException("array");
  5. if (index < 0 || index > array.Length)
  6. {
  7. throw new ArgumentOutOfRangeException("index", SR.GetString("IndexOutOfRange", new object[1]
  8. {
  9. (object) index
  10. }));
  11. }
  12. else
  13. {
  14. if (array.Length - index < this.Count)
  15. throw new ArgumentException(SR.GetString("Arg_InsufficientSpace"));
  16. LinkedListNode<T> linkedListNode = this.head;
  17. if (linkedListNode == null)
  18. return;
  19. do
  20. {
  21. array[index++] = linkedListNode.item;
  22. linkedListNode = linkedListNode.next;
  23. }
  24. while (linkedListNode != this.head);
  25. }
  26. }

以上。

.NET源码中的链表的更多相关文章

  1. ASP.NET MVC Filters 4种默认过滤器的使用【附示例】 数据库常见死锁原因及处理 .NET源码中的链表 多线程下C#如何保证线程安全? .net实现支付宝在线支付 彻头彻尾理解单例模式与多线程 App.Config详解及读写操作 判断客户端是iOS还是Android,判断是不是在微信浏览器打开

    ASP.NET MVC Filters 4种默认过滤器的使用[附示例]   过滤器(Filters)的出现使得我们可以在ASP.NET MVC程序里更好的控制浏览器请求过来的URL,不是每个请求都会响 ...

  2. Android 源码中的设计模式

    最近看了一些android的源码,发现设计模式无处不在啊!感觉有点乱,于是决定要把设计模式好好梳理一下,于是有了这篇文章. 面向对象的六大原则 单一职责原则 所谓职责是指类变化的原因.如果一个类有多于 ...

  3. 从express源码中探析其路由机制

    引言 在web开发中,一个简化的处理流程就是:客户端发起请求,然后服务端进行处理,最后返回相关数据.不管对于哪种语言哪种框架,除去细节的处理,简化后的模型都是一样的.客户端要发起请求,首先需要一个标识 ...

  4. Android 网络框架之Retrofit2使用详解及从源码中解析原理

    就目前来说Retrofit2使用的已相当的广泛,那么我们先来了解下两个问题: 1 . 什么是Retrofit? Retrofit是针对于Android/Java的.基于okHttp的.一种轻量级且安全 ...

  5. Eclipse与Android源码中ProGuard工具的使用

    由于工作需要,这两天和同事在研究android下面的ProGuard工具的使用,通过查看android官网对该工具的介绍以及网络上其它相关资料,再加上自己的亲手实践,算是有了一个基本了解.下面将自己的 ...

  6. String源码中的"avoid getfield opcode"

    引言: 之前一篇文章梳理了String的不变性原则,还提到了一段源码中注释"avoid getfield opcode",当时通过查阅资料发现,这是为了防止 getfield(获取 ...

  7. android源码中修改wifi热点默认始终开启

    在项目\frameworks\base\wifi\java\android\net\wifi\WifiStateMachine.java里面,有如下的代码,是设置wifi热点保持状态的:如下: pri ...

  8. rxjava源码中的线程知识

    rxjava源码中的线程知识 rx的最精简的总结就是:异步 这里说一下以下的五个类 1.Future2.ConcurrentLinkedQueue3.volatile关键字4.AtomicRefere ...

  9. MMS源码中异步处理简析

    1,信息数据的查询,删除使用AsycnQueryHandler处理 AsycnQueryHandler继承了Handler public abstract class AsyncQueryHandle ...

随机推荐

  1. sublime使用技巧汇总

    sublime使用技巧 Ubuntu下安装sublime text 3143版本 Install the GPG key: wget -qO - https://download.sublimetex ...

  2. 循环队列 & 栈的共用空间

    循环队列 非常好的数据结构,充分利用率空间,可以用于网络端存储socket消息! /*************************************** 作者: 未闻花语 版本: v1.0 ...

  3. 利用Go2Shell 实现 Mac Finder 直接shell端打开当前文件夹

    Finder 窗口 ,点击下图所示的按钮(红色框内),即可打开Shell Terminal. 打开后,如图 用法 安装go2shell后,打开finder的application文件夹,找到go2sh ...

  4. 谈谈最近的想法和 Thoughtworks 的 Offer

    最近笔者一直没有记录博客,原因是因为卷入了面试,离职,谈判,思考等一系列事件中.不过可以先说明一下的是, 笔者最后还是拒绝了 Thoughtworks 的 Offer,继续留在目前的公司. 去年毕业后 ...

  5. MySQL 基础--时间戳类型

    时间戳数据存储 .TimeStamp的取值范围为'1970-01-01 00:00:01' UTC 至'2038-01-19 03:14:07' UTC: .在存储时间戳数据时先将数据转换为UTC时区 ...

  6. Postgres通用翻页函数

    CREATE OR REPLACE FUNCTION fun_turnpage( PageSize INT, PageIndex INT, FldSort VARCHAR, StrCondition ...

  7. zookeeper日志级别

    查看源代码得知zookeeper(版本3.4.13)内部的日志用的slf4j,项目启动zk连接了之后一直在打debug日志(如下所示),甚是讨厌,logback日志级别调成info没用. 17:24: ...

  8. elasticsearch学习之根据发布时间设置衰减函数

    衰减函数decay functionion 高斯衰减 GET _search { "_source": ["title","release_date& ...

  9. web API简介(一):API,Ajax和Fetch

    概述 今天逛MDN,无意中看到了web API简介,觉得挺有意思的,就认真读了一下. 下面是我在读的时候对感兴趣的东西的总结,供自己开发时参考,相信对其他人也有用. 什么是API API (Appli ...

  10. [CocoaPods]如何使用CocoaPods插件

    CocoaPods +插件 CocoaPods是一个由极少数维护者运营的社区项目,需要维护大量的表面区域.可以肯定地说CocoaPods永远不会支持Xcode支持的每个功能,即使这样,团队也必须对许多 ...