数据结构 - 循环链表

单向循环链表

image.png
const ELEMENT_NOT_FOUND = -1;

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

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');
        }
    }
}

// 循环链表是一个环形,与普通的链表只是在add和remove上有不同,而且只是在向头部添加时不同
class CircleLinkedList 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) {
            // 先创建一个元素
            let node = new Node(ele, this.first);
            // 如果当前没有元素时,last就为创建的元素
            let last = this.size === 0 ? node : this.node(this.size - 1);
            // 讲last的next指向第一个新创建的第一个元素
            last.next = node;
            this.first = node;
        } else {
            let prevNode = this.node(index - 1);
            prevNode.next = new Node(ele, prevNode.next);
        }
        this.size++;
    }

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

        if (index === 0) {
            if (this.size === 1) {
                // 如果只剩一个元素需要直接置为null
                this.first = null;
            } else {
                // 如果剩多个元素时,先删除元素
                this.first = this.first.next;
                // 再将最后一个元素的next指向first第一个元素
                let last = this.node(this.size - 1);
                last.next = this.first;
            }
        } else {
            // 将前一个next指向下一个即可
            let prevNode = this.node(index - 1);
            let 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;

        //TODO 循环链表中不可以使用这种循环,需要使用for循环
        while (!!node) {
            if (ele === node.element) return index;

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

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

        //TODO 循环链表中不可以使用这种循环,需要使用for循环
        while (!!node) {
            str += `${node.element}-->`;
            node = node.next;
        }
        str += `size = ${this.size}`
        return str;
    }
}
©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

相关阅读更多精彩内容

  • 循环链表 链表的两头连接,形成了一个环状链表,称为循环链表。 实际应用:约瑟夫环问题 约瑟夫环问题,是一个经典的循...
    飞扬code阅读 22,906评论 1 7
  • 单链表的局限 单链表可以用于表示任意的线性关系,但有些线性关系是循环的,即没有队尾元素的 循环链表的定义 将单链表...
    Lin_ZJ阅读 729评论 0 5
  • Java 数据结构 循环链表 简介 循环链表与前两篇文章所提及的单向链表及双向链表也并没有太多不同的地方,只是其尾...
    Sheldonlv阅读 361评论 0 0
  • 牌的起始位置是如何摆放的?如何利用循环链表来解决? 核心算法
    豆瓣奶茶阅读 267评论 0 0
  • 近日随着自己干的事情越来越多,对自己有了一份不一样的觉察。 我发现我会很害怕听见别人的认可与表扬。 曾今的一段时间...
    沫沫杂谈阅读 132评论 0 1

友情链接更多精彩内容