数据结构 - 单向链表

image.png
const ELEMENT_NOT_FOUND = -1;

class BaseList {
    constructor() {
        this.size = 0;
    }

    // 获取size
    getSize() {
        return this.size;
    }

    // list是否为空
    isEmpty() {
        return this.size === 0;
    }

    // 是否包含一个元素
    contains(ele) {
        return this.indexof(ele) !== ELEMENT_NOT_FOUND;
    }

    // 检查是否超出范围
    rangeCheck(index, isadd) {
        if (isadd) {
            if (index > this.size || index < 0) throw new RangeError('index is out of range');
        } else {
            if (index >= this.size || index < 0) throw new RangeError('index is out of range');
        }
    }
}

class Node {
    constructor(ele, next) {
        this.element = ele;
        this.next = next;
    }
}

class LinkList extends BaseList {
    constructor() {
        super();
        this.first = null;
    }

    // 清空
    clear() {
        this.first = null;
        this.size = 0;
    }

    // 获取固定下标的值 复杂度: O(n)
    get(index) {
        return this.node(index).element;
    }

    // 设置固定下标的值 复杂度: O(n)
    set(index, ele) {
        let node = this.node(index);
        let old = node.element;
        node.element = ele;
        return old;
    }

    // 向末尾增加 复杂度: O(n)
    push(ele) {
        this.add(ele, this.size);
    }

    // 添加元素 复杂度: O(n)
    add(ele, index) {
        this.rangeCheck();

        // 要区分向头部添加和其他位置添加的情况
        if (index === 0) {
            this.first = new Node(ele, this.first);
        } else {
            let prevNode = this.node(index - 1);
            prevNode.next = new Node(ele, prevNode.next);
        }
        this.size++;
    }

    // 删除元素 复杂度: O(n)
    remove(index) {
        this.rangeCheck();

        let currentNode = this.first;
        if (index === 0) {
            this.first = this.first.next;
        } else {
            // 将前一个next指向下一个即可
            let prevNode = this.node(index - 1);
            currentNode = prevNode.next;
            prevNode.next = currentNode.next;
        }
        this.size--;

        return currentNode.element;
    }

    // 获取当前下标的元素 复杂度: O(n)
    node(index) {
        this.rangeCheck(index);

        let node = this.first;
        for (let i = 0; i < index; i++) {
            node = node.next;
        }
        return node;
    }

    // 获取元素的下标 复杂度: O(n)
    indexof(ele) {
        let node = this.first,
            index = 0;

        while (!!node) {
            if (ele === node.element) return index;

            node = node.next;
            index++
        }
        return ELEMENT_NOT_FOUND;
    }

    toString() {
        let node = this.first,
            str = '';

        while (!!node) {
            str += `${node.element}-->`;
            node = node.next;
        }
        str += `size = ${this.size}`
        return str;
    }
}

单向链表优点:

  1. 不会造成内存空间的浪费,需要多少开辟多少。
    单向链表缺点:
  2. 会频繁的开辟、删除内存空间。

由于添加、删除链表元素都需要判断是否是操作第一个节点,可以采用增加虚拟头节点的方式来统一处理

image.png

// 虚拟头节点的链表
class Linklist extends BaseList {
    constructor() {
        super();
        // 增加虚拟头节点 -- 方便之后统一处理
        this.first = new Node(null, null);
    }

    clear() {
        this.first = null;
        this.size = 0;
    }

    get(index) {
        return this.node(index).element;
    }

    set(index, ele) {
        let node = this.node(index);
        let old = node.element;
        node.element = ele;
        return old;
    }

    push(ele) {
        this.add(ele, this.size);
    }

    add(ele, index) {
        this.rangeCheck();

        // if (index === 0) {
        //     this.first = new Node(ele, this.first);
        // } else {
        //     let prevNode = this.node(index - 1);
        //     prevNode.next = new Node(ele, prevNode.next);
        // }

        `统一处理即可,不需要再进行判断`
        let prevNode = index === 0 ? this.first : this.node(index - 1);
        prevNode.next = new Node(ele, prevNode.next);

        this.size++;
    }

    remove(index) {
        this.rangeCheck();

        // if (index === 0) {
        //     this.first = this.first.next;
        // } else {
        //     let prevNode = this.node(index - 1);
        //     let currentNode = prevNode.next;
        //     prevNode.next = currentNode.next;
        // }

        `统一处理`
        let prevNode = index === 0 ? this.first : this.node(index - 1);
        let currentNode = prevNode.next;
        prevNode.next = currentNode.next;

        this.size--;

        return currentNode.element;
    }

    node(index) {
        this.rangeCheck(index);

        // 循环的第一个一定是从虚拟节点的next开始
        let node = this.first.next;
        for (let i = 0; i < index; i++) {
            node = node.next;
        }
        return node;
    }

    indexof(ele) {
        // 循环的第一个一定是从虚拟节点的next开始
        let node = this.first.next,
            index = 0;

        while (!!node) {
            if (ele === node.element) return index;

            node = node.next;
            index++
        }
        return ELEMENT_NOT_FOUND;
    }

    toString() {
        // 循环的第一个一定是从虚拟节点的next开始
        let node = this.first.next,
            str = '';

        while (!!node) {
            str += `${node.element}-->`;
            node = node.next;
        }
        str += `size = ${this.size}`
        return str;
    }
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • C语言中,我们在使用数组时,会需要对数组进行插入和删除的操作,这时就需要移动大量的数组元素,但在C语言中,数组属于...
    正义的程序员阅读 9,511评论 0 3
  • 1. 链表概述 链表是有序的列表,但是它在内存中是存储如下 链表是以节点的方式来存储,是链式存储 每个节点包含 d...
    21号新秀_邓肯阅读 2,750评论 0 0
  • Java 数据结构 单向链表 基础介绍 链表与循序表都是同属于数据结构中顺序表中的一种,而它与循序表的不同就在于 ...
    Sheldonlv阅读 2,241评论 0 0
  • 定义 单向链表(单链表)是链表的一种,其特点是链表的链接方向是单向的,对链表的访问要通过顺序读取从头部开始;链表是...
    再见噜噜班阅读 754评论 0 0
  • 单向链表与数组备用图示 带头单链表插入新的节点图示 带头双向链表 带头双向链表插入 参考Code LinkNode...
    欣_可期阅读 681评论 0 1