数据结构——链表

  • 源码地址请点击此处

链表也是一种常见的数据结构,从数据结构类型上区分,链表属于存储结构的一种:链式存储结构。
和顺序存储结构一样,链式存储结构也可以用来作为实现其他数据结构的基础,比如前面介绍的线性表、栈、队列等。

顺序存储结构和链式存储结构的区别

使用顺序存储结构存放数据时,使用的是内存空间中连续的内存,如下图所示:


数据结构——链表_第1张图片
连续内存空间

使用顺序存储结构时,数据需要存放在连续的内存空间中。这种存储结构的优点在于读取速度很快,缺点在于无法有效利用内存空间。怎么理解呢?举个例子,如果只使用顺序存储结构,在程序的运行过程中,会产生很多的临时变量,这些变量在不用时会被垃圾回收,当变量被回收后,内存中就出现了不连续的情况,也可以叫做内存空洞。如下图所示,黑色区域表示内存空洞:


数据结构——链表_第2张图片
内存空洞

随着程序的运行,这样的内存空洞会越来越多,此时如果程序中需要用到较大的内存空间,而剩余的连续内存不够的情况下,将无法成功分配内存,但内存中又有很多的内存空洞,却无法得到有效利用。
要解决这个问题,有两个解决方案:
  1. 将内存空洞后面的数据往前移
    第一种方式就是每隔一段时间,将内存空洞后面的数据向前移,以达到“挤压”内存空洞的目的。这种方式的缺点在于性能浪费严重。
  2. 使用链式存储结构
    第二种方式就是使用链式存储结构:在某些内存空洞中存放一部分数据,再存放下一块可用内存空洞的地址,通过地址把这些可用的内存空洞连接起来,实现一个链式的结构,这就是链表。下面是链表的基本结构:


    数据结构——链表_第3张图片
    链表

    上图展示了链表的基本结构,对于链表,其中的每个元素我们称之为节点。可见每个节点至少需要包含两部分内容:

  • 数据区:用来存放数据
  • 地址区:用来存放下一个节点的地址

链表的分类

通常我们说的链表也叫作单向链表,此外,链表还有两个常用的衍生结构:双向链表和循环链表。本文主要介绍单向链表的实现,双向链表和循环链表在后文介绍。

链表的代码实现

下面是链表的代码实现,首先来定义一个 ISingleNode 接口和 SingleNode 类,用来实现链表中的节点。

节点类和接口的代码实现

interface ISingleNode{
    // 数据区
    data:T;
    // 地址区
    next:ISingleNode;
}

class SingleNode implements ISingleNode{
    data:T = null;
    next:ISingleNode = null;
    constructor(data?:T){
        this.data = data;
    }
}

链表类的接口和代码实现

首先定义 ISingleLinkedList 接口,规范链表类中的基本操作方法:

interface ISingleLinkedList{
    // 获取链表的长度
    size():number;
    // 获取链表头
    head():ISingleNode;
    // 增加节点
    append(item:T):ISingleNode;
    // 删除节点
    remove(item:T):void;
    // 根据位置删除
    removeAt(pos:number):void;
    // 插入节点
    insert(newItem:T,oldItem:T):ISingleNode;
    // 在具体的位置插入
    insertAt(newItem:T,pos:number):ISingleNode;
    // 清空链表
    clear():void;
    // 判断链表是否为空
    isEmpty():boolean;
    // 查找节点和其前驱
    find(item:T):{
        previous:ISingleNode,
        current:ISingleNode,
        currentPos:number,
        previousPos:number
    };
    // 根据位置查找节点和其前驱
    findAt(pos:number):{
        previous:ISingleNode,
        current:ISingleNode,
        currentPos:number,
        previousPos:number
    };
    // 获取链表中的元素
    toString():ISingleNode[];
}

定义链表类 SingleLinkedList

class SingleLinkedList implements ISingleLinkedList{
    private _size:number = 0;
    private _head:ISingleNode = new SingleNode();
}

下面逐个讲解链表类中的几个关键方法。

  1. 实现 find() 方法
    首先需要实现 find() 方法,该方法用来从链表中查找节点,以供后面的添加和删除方法使用。
find(item:T):{
    previous:ISingleNode,
    current:ISingleNode,
    currentPos:number,
    previousPos:number,
}{
    if(!item){
        throw new Error("参数错误!")
    }
    let 
        previous:ISingleNode = null,
        current:ISingleNode = this._head,
        index:number = -1;
    while(current){
        // 更新索引值
        index++;
        // 判断当前节点中的数据和传入的是否匹配
        if(current.data === item){
            break;
        }
        // 将 current 赋值给 previous
        // 将 current.next 赋值给 current
        // 在下一次迭代中使用
        previous = current;
        current = current.next;
    }

    // HACK 在前面的循环中找不到对于的元素时,会获取到尾节点
    // 这里进行一次二次验证
    if(current.data !== item){
        index = -1;
    }

    // 处理未找到的情况
    if(index === -1){
        return{
            previous:null,
            current:null,
            previousPos:-1,
            currentPos:-1
        }
    }
    return{
        previous,
        current,
        currentPos:index,
        // 前驱的位置在当前节点之前
        previousPos:index - 1
    }
}

这个方法(和配套的 findAt())是链表类中核心方法,再进行链表的添加、插入或删除操作时,首先需要查找链表中的某个节点和其前驱节点,查找总是从链表的头部开始的。

  1. 实现 append() 方法
    该方法用来向链表中添加节点,在添加时有两种情况:
  • 链表为空时,将头节点替换为新节点
  • 链表不为空时,将尾结点替换为新节点
append(item:T):ISingleNode{
    const newNode = new SingleNode(item);
    // 链表中没有节点
    if(!this._size){
        this._head = newNode;
    }else{
        const {current} = this.findAt(this._size - 1);
        current.next = newNode
    }
    this._size++;
    return newNode;
}

3.实现 insert() 方法
该方法用来向链表中插入元素,也有两种情况:

  • 插入的位置是头节点,将头节点替换为新节点
  • 插入的位置是其他节点,将原始节点的位置插入新节点,同时将原始节点向后移,这时就需要用到原始节点的前驱节点
insert(newItem:T,oldItem:T):ISingleNode{
    // 创建新节点
    const newNode = new SingleNode(newItem);
    // 查找旧节点及其前驱节点
    const {current,previous} = this.find(oldItem);
    // 没有查找到旧节点,直接返回
    if(!current) return null;
    // 当 previous 为 null 时,说明是头节点
    if(!previous){
        newNode.next = current;
        this._head = newNode;
    }else{
        // 将新建节点的 next 指向旧节点
        newNode.next = current;
        // 将旧节点前驱的 next 指向新建的节点
        previous.next = newNode;
    }
    this._size++;
    return newNode;
}
  1. 实现 remove() 方法
    该方法用来从链表中移除节点。
remove(item:T):void{
    // 获取当前节点和其的前驱
    let { current,previous } = this.find(item);
    // 还没有添加节点的情况
    if(!current) return;
    // 没有前驱节点,说明是头节点
    if(!previous){
        this._head = current.next;
    }else{
        // 将当前节点的前驱的 next 指向当前节点的后继
        previous.next = current.next;
    }
    // 移除当前节点
    current = null;
    // 更新链表长度
    this._size--;
}

下面是 SingleLinkedList 类的完整代码实现:

class SingleLinkedList implements ISingleLinkedList{
    private _size:number = 0;
    private _head:ISingleNode = new SingleNode();
    size():number{
        return this._size;
    }
    head():ISingleNode{
        return this._head;
    }
    clear():void{
        this._head = null;
        this._head = new SingleNode();
        this._size = 0;
    }
    isEmpty():boolean{
        return !this._size;
    }
    append(item:T):ISingleNode{
        const newNode = new SingleNode(item);
        // 链表中没有节点
        if(!this._size){
            this._head = newNode;
        }else{
            const {current} = this.findAt(this._size - 1);
            current.next = newNode
        }
        this._size++;
        return newNode;
    }
    find(item:T):{
        previous:ISingleNode,
        current:ISingleNode,
        currentPos:number,
        previousPos:number,
    }{
        if(!item){
            throw new Error("参数错误!")
        }
        let 
            previous:ISingleNode = null,
            current:ISingleNode = this._head,
            index:number = -1;
        while(current){
            // 更新索引值
            index++;
            // 判断当前节点中的数据和传入的是否匹配
            if(current.data === item){
                break;
            }
            // 将 current 赋值给 previous
            // 将 current.next 赋值给 current
            // 在下一次迭代中使用
            previous = current;
            current = current.next;
        }

        // HACK 在前面的循环中找不到对于的元素时,会获取到尾节点
        // 这里进行一次二次验证
        if(current.data !== item){
            index = -1;
        }

        // 处理未找到的情况
        if(index === -1){
            return{
                previous:null,
                current:null,
                previousPos:-1,
                currentPos:-1
            }
        }
        return{
            previous,
            current,
            currentPos:index,
            // 前驱的位置在当前节点之前
            previousPos:index - 1
        }
    }
    findAt(pos:number):{
        previous:ISingleNode,
        current:ISingleNode,
        currentPos:number,
        previousPos:number,
    }{
        let 
            previous:ISingleNode = null,
            current:ISingleNode = this._head,
            index:number = -1;
            
        if(pos < 0 || pos > this._size - 1){
            throw  new Error("索引越界!");
        }

        while(current){
            index++;
            if(index === pos){
                break;
            }
            previous = current;
            current = current.next;
        }

        // 处理未找到的情况
        if(index === -1){
            return{
                previous:null,
                current:null,
                previousPos:-1,
                currentPos:-1
            }
        }
        return{
            previous,
            current,
            currentPos:index,
            // 前驱的位置在当前节点之前
            previousPos:index - 1
        }
    }
    remove(item:T):void{
        // 获取当前节点和其的前驱
        let { current,previous } = this.find(item);
        // 还没有添加节点的情况
        if(!current) return;
        // 没有前驱节点,说明是头节点
        if(!previous){
            this._head = current.next;
        }else{
            // 将当前节点的前驱的 next 指向当前节点的后继
            previous.next = current.next;
        }
        // 移除当前节点
        current = null;
        // 更新链表长度
        this._size--;
    }    
    removeAt(pos:number):void{
        // 获取当前节点和其的前驱
        let { current,previous } = this.findAt(pos);
        // 还没有添加节点的情况
        if(!current) return;
        // 没有前驱节点,说明是头节点
        if(!previous){
            this._head = current.next;
        }else{
            // 将当前节点的前驱的 next 指向当前节点的后继
            previous.next = current.next;
        }
        // 移除当前节点
        current = null;
        // 更新链表长度
        this._size--;
    }
    insert(newItem:T,oldItem:T):ISingleNode{
        // 创建新节点
        const newNode = new SingleNode(newItem);
        // 查找旧节点及其前驱节点
        const {current,previous} = this.find(oldItem);
        // 没有查找到旧节点,直接返回
        if(!current) return null;
        // 当 previous 为 null 时,说明是头节点
        if(!previous){
            newNode.next = current;
            this._head = newNode;
        }else{
            // 将新建节点的 next 指向旧节点
            newNode.next = current;
            // 将旧节点前驱的 next 指向新建的节点
            previous.next = newNode;
        }
        this._size++;
        return newNode;
    }
    insertAt(newItem:T,pos:number):ISingleNode{
        // 创建新节点
        const newNode = new SingleNode(newItem);
        // 查找旧节点及其前驱节点
        const {current,previous} = this.findAt(pos);
        if(!current) return null;
        // 当 previous 为 null 时,说明是头节点
        if(!previous){
            newNode.next = current;
            this._head = newNode;
        }else{
            // 将新建节点的 next 指向旧节点
            newNode.next = current;
            // 将旧节点前驱的 next 指向新建的节点
            previous.next = newNode;
        }
        this._size++;
        return newNode;
    }
    toString():ISingleNode[]{
        const tmp:ISingleNode[] = [];
        let current:ISingleNode = this._head;
        while(current){
            tmp.push(current);
            current = current.next;
        }
        return tmp;
    }
}

总结

链表在实现上比前面几个数据结构要复杂一点,这是因为在实现链表的添加、插入和删除操作时还需要修复节点地址区的指向问题。通常,链表操作需要考虑两种情况:

  • 在头节点进行添加、插入和删除操作
  • 在其他位置进行添加、插入和删除操作

当在头节点进行操作时,不需要使用头节点的前驱节点,因为链表的头节点没有前驱节点,只需要修正头节点的 next 属性,或者使用新的节点替换头节点。
在其他位置进行操作时,需要使用到该节点的前驱节点,并修正前驱节点的 next 属性。同时,如果是删除操作,还需要使用到该节点的后继节点,在修正前驱节点和后继节点的地址区指向后,还需要将被删除的节点置为 null,防止内存泄露。

完。

你可能感兴趣的:(数据结构——链表)