美文网首页
数据结构 - 单向链表

数据结构 - 单向链表

作者: Super曲江龙Kimi | 来源:发表于2020-03-03 20:54 被阅读0次
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;
    }
}

相关文章

  • 算法与数据结构知识汇总(二、链表)

    1、概念 2、链表的数据结构 单向链表的数据结构如下图: 上图数据结构为单向链表,简称单链表,该数据结构由若干个节...

  • 2019-12-04 Java-LinkedList源码解读

    @TOC 1、链表数据结构 链表分为单向链表和双向链表,他们的区别在于,单向链表只能单向寻址,而双向链表可以双向寻...

  • 用Java写单向链表

    数据结构—单向链表 为了巩固自己的基础知识,这次就用 Java 来写一个单向链表。问:什么是单向链表?首先链表是数...

  • 数据结构 | 其二 链表

    冰河winner - 数据结构之链表 2.1 单向链表 数据结构(一) 单链表的实现-JAVA 2.2 双端链表 ...

  • 线性表-单向循环链表

    单向循环链表 单向循环链表示意图如下: 数据结构定义(同普通链表) 单向循环链表初始化与赋值 在上面循环遍历查找尾...

  • 数据结构笔记

    数据结构课程概览 ================== 1.顺序表 2.链表:单链表,单向循环链表,双链表...

  • 链表反转

    概述 链表反转是非常经典的面试题,要实现此功能,需先实现链表的数据结构。 链表类 获得单向链表方法 输出单向链表方...

  • 总结

    Android篇 数据结构与算法顺序表 - ArrayList源码链表 - 单向链表、双向链表 - LinkedL...

  • 2018-03-26

    数据结构:单向链表 #include#include//实现一个简单的单向链表 (不带头节点的链表,只有一个头指针...

  • 链表

    一种线性数据结构。包含单向链表和双向链表。 单向链表的结构,操作(插入、删除和遍历)及其时间复杂度: 通常使...

网友评论

      本文标题:数据结构 - 单向链表

      本文链接:https://www.haomeiwen.com/subject/iudllhtx.html