数据结构之LinkList链表

链表与数组在数据结构的江湖上被并称为南数组、北链表,其江湖地位可见一斑

概念

链表作为最基础的通用存储结构,它的作用和数组是一样的,但存储数据的方式略有不同。数组需要预先获取一定的内存空间,且内存的地址是连续的,在存储数据的时候,插入和删除都需要遍历数组,还需要移动部分数据,是十分低效的。而链表能够解决以上问题,下面来看一下链表的结构。

链表的数据项存储在链表节点中,一个节点可以叫Node也可以叫Link,它包含数据项和一个引用,数据项存储数据本身,通常是一个包含各种数据的类的对象,而引用则是引用下一个节点,一般记为next,链表的核心实现就是这个next,它标记了每个节点之间的联系,像一根线一样把这些散落在内存各个角落的节点对象串了起来。下面通过代码来直观的理解一下:

/**
 * 链表的节点,用来存放数据
 */
public class Node {
    // 数据变量
    private String data;
    // next引用
    public Node next;
    // 构造函数
    public Node(String str) {
        data = str;
    }
    // getter
    public String getData() {
        return data;
    }
    // Node的打印方法
    public void displayNode() {
    System.out.print("{ Node: " + data + " }..");
    }
}

可以看到,链表节点的关键两个属性,一个是数据项(这里简单的表示为String类型的data),一个是指向下一个链表节点的next引用(这里为了访问简单设置为public),虽然还不知道下一个节点是谁,但在链表的数据数据结构中,将数据插入时,就会建立这个联系。

单向链表

以上说明了链表的重要概念,节点与引用,下面来展示如何构成一个链表,首先从最简单的单向链表开始,单向链表顾名思义,只有一端可以插入和访问的链表,链表类LinkList值仅维护第一个链表节点first的信息,其它的节点通过next引用,以下是实现代码:

/**
 * 单向链表:维护头节点的信息,对链表的操作都从头部开始
 */
public class LinkList {
    // 头节点的引用
    Node first;
    // 构造方法
    public LinkList() {
        first = null;
    }
    // 插入方法,头插法
    public void insertFirst(String str) {
        Node newNode = new Node(str);
        newNode.next = first;
        first = newNode;
    }
    // 删除方法,删除头部节点
    public Node deleteFirst() {
        Node temp = first;
        first = first.next;
        return temp;
    }
    // 打印链表方法
    public void display() {
        System.out.print("LinkList: ");
        Node current = first;
        while (current != null) {
            current.displayNode();
            current = current.next;
        }
        System.out.println();
    }
    // 判断链表是否为空的方法
    public Boolean isEmpty() {
        return first == null;
    }
    // 在链表中查找的方法
    public Node find(String str) {
        Node current = first;
        while (current != null) {
            if (current.getData() != str)
                current = current.next;
            return current;
        }
        System.out.println("could not find Node :" + str);
        return null;
    }
    // 从链表中查找元素删除方法
    public Node delete(String str) {
        Node current = first;
        Node previous = first;
        while (current != null) {
            if (current.getData() != str) {
                previous = current;
                current = current.next;
            } else {
                previous.next = current.next;
                return current;
            }
        }
        System.out.println("cloud not find Node:" + str + ", failed to delete it");
        return null;
    }
}

代码中定义了唯一一个变量:头节点的引用first,还有一些链表的常用方法:插入、删除、查找、展示等,可以发现这些方法都是由头节点first展开的,在表达这些方法的时候需要注意边界和特殊情况first的分析。

双端链表

注意是双端列表,不是双向链表,在上面的单向链表中,链表类仅维护了一个头节点的引用first,那么如果要在链表尾部插入一个数据,需要遍历整个链表,这种效率很低。双端链表不仅维护了first头节点的引用,而且还维护了last尾节点的引用,这样访问尾节点就像访问头节点一样方便,这种特性使得双端链表比起单向链表在一些场合更有效率,比如说队列。

以下展示双端链表的实现代码,节点还是引用上面的Node类

public class DoubleSideLinkList {

    // 首节点和尾节点的引用
    Node first;
    Node last;

    // constructor
    public DoubleSideLinkList() {
        first = null;
        last = null;
    }

    // isEmpty
    public boolean isEmpty() {
        return first == null;
    }

    // insert first
    public void insertFirst(String str) {
        Node newNode = new Node(str);
        if (isEmpty()) {
            last = newNode;
        }
        newNode.next = first;
        first = newNode;
    }

    // insert last
    public void insertLast(String str) {
        Node newNode = new Node(str);
        if (isEmpty()) {
            first = newNode;
        }
        last.next = newNode;
        last = newNode;
    }

    // delete first
    public Node deleteFirst() {
        Node temp = first;
        // 仅有一个元素
        if (first.next == null)
            last = null;
        first = first.next;
        return temp;
    }

    // display
    public void displayList() {
        System.out.println("List (first-->last): ");
        Node current = first;
        while(current != null) {
            current.displayNode();
            current = current.next;
        }
        System.out.println();
    }
}

可以看出,双端链表可以很方便的从尾节点插入数据,这里需要说明的是,这个链表依然是单向的,每个节点也仅有一个引用next指向下一个节点,所以遍历和查找的方法与单向链表没什么差别,在此就不做展示。在实现双端链表的insert与delete方法的时候需要注意特殊情况:列表为空或者列表将要为空时,first与last的处理。

有序链表

有序链表中,数据是按照关键值的有序排列的,有序链表的插入速度优于有序数组,而且可以扩展到所有有效的内存,所以在很多场合可以使用有序链表,在以下的实现中,我们还是引用上面的Node类,因为数据项data为String类型,我们以数据项的hashcode为序来实现有序链表:

public class SortedLinkList {

    // filed
    Node first;

    // constructor
    public SortedLinkList() {
        first = null;
    }

    public boolean isEmpty() {
        return (first==null);
    }

    public void insert(String key) {
        Node current = first;
        Node previous = null;
        Node newNode = new Node(key);

        while (current != null && current.getData().hashCode() < key.hashCode()) {
            previous = current;
            current = current.next;
        }
        if (previous == null) {
            first = newNode;
        } else {
            previous.next = newNode;
        }
        newNode.next = current;
    }

    public Node remove() {
        Node temp = first;
        first = first.next;
        return temp;
    }

    public void display() {
        System.out.print("LinkList: ");
        Node current = first;
        while (current != null) {
            current.displayNode();
            current = current.next;
        }
        System.out.println();
    }
}

有序链表主要是insert方法比较难实现,也是需要考虑特殊情况,即链表为空。

双向链表

双向链表与双端链表不同,双端链表虽然可以从尾部插入节点,但是每个节点引用的还是其下一个节点,遍历也是单向的,而双向链表则可以从头部或者从尾部开始遍历,为了实现这一特性,我们需要修改一下节点类Node,为了区分,我们以Link为节点命名,在Link中,不仅实现对下一节点的引用next,而且还实现了对上一个节点的引用previous,如下:

public class Link {

    private int data;

    public Link next;

    public Link previous;

    public Link(int data) {
        this.data = data;
    }

    public int getData() {
        return data;
    }

    public void displayLink() {
        System.out.print(data + " ");
    }
}

唯一不同的点就是指向上一个节点的引用previous,虽然仅在节点中加了一个引用,但是双向链表的实现却麻烦了很多,其中各种引用关系特别容易混乱,而且还有一些特殊情况需要考虑,下面一起来感受一下:

public class DoublyLinkList {

    // 头节点和尾节点的引用,也是双端的
    Link first;
    Link last;

    // constructor
    public DoublyLinkList() {
        first = null;
        last = null;
    }

    // isEmpty
    public boolean isEmpty() {
        return first == null;
    }

    // insert first
    public void insertFirst(int data) {
        Link newLink = new Link(data);

        if (isEmpty()) {
            last = newLink;
        } else {
            first.previous = newLink;
        }
        newLink.next = first;
        first = newLink;
    }

    // insert last
    public void insertLast(int data) {
        Link newLink = new Link(data);

        if (isEmpty()) {
            first = newLink;
        } else {
            last.next = newLink;
            newLink.previous = last;
        }
        last = newLink;
    }

    // insert after
    public boolean insertAfter(int key, int data) {
        Link newLink = new Link(data);
        Link current = first;
        // find
        while(current != null) {
            if (current.getData() != key) {
                current = current.next;
            } else {
                if (current == last) {
                    newLink.next = null;
                    last = newLink;
                } else {
                    newLink.next = current.next;
                    current.next.previous = newLink;
                }
                current.next = newLink;
                newLink.previous = current;
                return true;
            }
        }
        System.out.println("cloud not find the key: " + data);
        return false;
    }

    // delete first
    public Link deleteFirst() {
        Link temp = first;
        if (first.next == null) {
            last = null;
        } else {
            first.next.previous = null;
        }
        first = first.next;
        return temp;
    }

    // delete last
    public Link deleteLast() {
        Link temp = last;
        if (first.next == null) {
            first = null;
        } else {
            last.previous.next = null;
        }
        last = last.previous;
        return temp;
    }

    // delete key
    public Link deleteKey(int data) {
        Link current = first;
        while (current != null) {
            if (current.getData() != data) {
                current = current.next;
            } else {
                if (current == first) {
                    first = first.next;
                } else {
                    current.previous.next = current.next;
                }
                if (current == last) {
                    last = last.previous;
                } else {
                    current.next.previous = current.previous;
                }
                return current;
            }
        }
        System.out.println("cloud not find the key: " + data + ", delete failed!");
        return null;
    }

    // display forward
    public void displayForward() {
        System.out.println("List (first-->last): ");
        Link current = first;
        while (current != null) {
            current.displayLink();
            current = current.next;
        }
        System.out.println();
    }

    // display backward
    public void displayBackward() {
        System.out.println("List (last-->first): ");
        Link current = last;
        while (current != null) {
            current.displayLink();
            current = current.previous;
        }
        System.out.println();
    }
}

这是一个双向链表,同时也是双端列表,可以从前向后或者从后向前遍历,也可以在链表的任何地方插入或是删除节点,在实现这些方法时,需要修改与插入或删除节点有关系的前后四个引用,且还需要考虑first与last、列表为空或者仅有一个元素的特殊情况,可以通过画图来加深理解。

效率与总结

以上我们介绍并实现了单向链表、双端链表、有序链表和双向链表,接下来我们来讨论一下这些链表的效率。
链表与数组作为通用数据结构经常被拿来做对比,以链表和数组这两个通用数据结构为基础的一些抽象数据模型,比如java中的ArrayList与LinkedList,也经常被拿出来作比较,总的来说主要有两大区别:

  1. 链表插入和删除的效率高于数组,链表插入、删除头节点、尾节点的时间效率为O(1),而在链表中间插入、删某一节点的时间复杂度为O(N),虽然与数组相同操作的时间复杂度相同,可是数组需要复制移动元素位置来填补空洞,所以链表的效率还是要优于数组的,特别是复制时间占比较重的情况中。
    数组则是随机访问元素的效率要高于链表,可以通过数组下标直接访问的到。

  2. 链表比数组优越的另一个重要因素就是,链表可以扩展到所有可用的内存空间,且不用像数组一样初始化容量,也不需要可以扩容。另外当一个数组对象需要一个连续的大内存空间时,会有几率触发jvm的GC去整理内存空间以获取一片连续的内存,而链表则没有这方面的限制,所以在内存的使用效率上,链表优于数组。

正如以上所述,一般来说,链表如果采用头插法(或删除),则时间复杂度均为O(1),而如果是随机插入或者向有序链表中插入的时间效率均为O(N),因为需要找到合适的位置;

有序链表可以在O(1)的时间返回或删除最小或最大值,如果一个应用需要频繁的返回最小值且不需要快速的插入时间,则可以选择有序链表来实现,如优先级队列;

双向链表由于两端都可以插入、删除和遍历,所以可以用来做双端队列。

参考文章

《Data Structures & Algorithms in Java》 Robert Lafore著

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 206,214评论 6 481
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 88,307评论 2 382
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 152,543评论 0 341
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 55,221评论 1 279
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 64,224评论 5 371
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,007评论 1 284
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,313评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,956评论 0 259
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,441评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,925评论 2 323
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,018评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,685评论 4 322
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,234评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,240评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,464评论 1 261
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,467评论 2 352
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,762评论 2 345

推荐阅读更多精彩内容

  • 一些概念 数据结构就是研究数据的逻辑结构和物理结构以及它们之间相互关系,并对这种结构定义相应的运算,而且确保经过这...
    Winterfell_Z阅读 5,654评论 0 13
  • ⦁ 数据结构 数据结构是计算机存储和组织数据的的方式 1. 数组 在Java中,数组是用来存放同一种数据类型的集合...
    欲火逢生阅读 511评论 0 3
  • 前面博客我们在讲解数组中,知道数组作为数据存储结构有一定的缺陷。在无序数组中,搜索性能差,在有序数组中,插入效率又...
    IT可乐阅读 601评论 0 2
  • 一、链表的定义 链表是一种递归的数据结构,是一种线性结构,但是并不会按线性的顺序存储数据,而是在每一个节点里存到下...
    熊喵先森阅读 1,464评论 0 3
  • (上)如何实现LRU缓存淘汰算法? 一、什么是链表? 1.和数组一样,链表也是一种线性表。2.从内存结构来看,链表...
    码语生活阅读 317评论 0 0