数据结构——双向链表、循环链表

本文介绍链表的两个衍生结构:双向链表和循环链表。

双向链表

前面介绍的单向链表中,每个节点的地址区域 next 会指向该节点的后继节点,在双向链表中,会再增加一个地址区域 prev,指向该节点的前驱节点。如下图所示:

双向链表

在双向链表中,每个节点需要增加一个 prev 属性,同时在对链表进行了添加、插入和删除操作后,需要同时修复 nextprev 的指向。
前面我们已经将链表最重要的查找方法 find()findAt() 实现了,这两个方法可以找到每个节点和其前驱节点,因此这里只需要稍微修改下添加、插入和删除方法,修复 prev 指向即可。

双向链表的代码实现

下面是双向链表的代码实现,首先实现 IDoubleNode 接口和 DoubleNode 类。

interface IDoubleNode<T>{
    // 数据区
    data:T;
    // 地址区
    // 前驱
    prev:IDoubleNode<T>;
    // 后继
    next:IDoubleNode<T>;
}

class IDoubleNode<T> implements IDoubleNode<T>{
    data:T = null;
    prev:IDoubleNode<T> = null;
    next:IDoubleNode<T> = null;
    constructor(data?:T){
        this.data = data;
    }
}

定义 IDoubleLinkedList 接口,该接口的定义和前面单向链表的接口一模一样,这里偷懒就直接搬过来了。如果您还有其他自定义的接口,可以自行实现:

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

下面是 DoubleLinkedList 类的实现,修改了添加、插入和删除的几个方法:

class DoubleLinkedList<T> implements IDoubleLinkedList<T>{
    private _size:number = 0;
    private _head:IDoubleNode<T> = new IDoubleNode();
    size():number{
        return this._size;
    }
    head():IDoubleNode<T>{
        return this._head;
    }
    clear():void{
        this._head = null;
        this._head = new IDoubleNode();
        this._size = 0;
    }
    isEmpty():boolean{
        return !this._size;
    }
    append(item:T):IDoubleNode<T>{
        const newNode = new IDoubleNode<T>(item);
        // 链表中没有节点
        if(!this._size){
            this._head = newNode;
        }else{
            const {current} = this.findAt(this._size - 1);
            current.next = newNode;
            // 将新节点的前驱指向最后一个节点
            newNode.prev = current;
        }
        this._size++;
        return newNode;
    }
    find(item:T):{
        previous:IDoubleNode<T>,
        current:IDoubleNode<T>,
        currentPos:number,
        previousPos:number,
    }{
        if(!item){
            throw new Error("参数错误!")
        }
        let 
            previous:IDoubleNode<T> = null,
            current:IDoubleNode<T> = 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:IDoubleNode<T>,
        current:IDoubleNode<T>,
        currentPos:number,
        previousPos:number,
    }{
        let 
            previous:IDoubleNode<T> = null,
            current:IDoubleNode<T> = 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;
            // 修正头节点的 prev 指向
            this._head.prev = null;
        }else{
            // 将当前节点的前驱的 next 指向当前节点的后继
            previous.next = current.next;
            // 设置当前节点的后一个节点的 prev 指向
            current.next.prev = previous;
        }
        // 移除当前节点
        current = null;
        // 更新链表长度
        this._size--;
    }    
    removeAt(pos:number):void{
        // 获取当前节点和其的前驱
        let { current,previous } = this.findAt(pos);
        // 还没有添加节点的情况
        if(!current) return;
        // 没有前驱节点,说明是头节点
        if(!previous){
            this._head = current.next;
            // 修正头节点的 prev 指向
            this._head.prev = null;
        }else{
            // 将当前节点的前驱的 next 指向当前节点的后继
            previous.next = current.next;
            // 设置当前节点的后一个节点的 prev 指向
            current.next.prev = previous;
        }
        // 移除当前节点
        current = null;
        // 更新链表长度
        this._size--;
    }
    insert(newItem:T,oldItem:T):IDoubleNode<T>{
        // 创建新节点
        const newNode = new IDoubleNode(newItem);
        // 查找旧节点及其前驱节点
        const {current,previous} = this.find(oldItem);
        // 没有查找到旧节点,直接返回
        if(!current) return null;
        // 当 previous 为 null 时,说明是头节点
        if(!previous){
            newNode.next = current;
            // 设置当前节点的 prev
            current.prev = newNode;
            this._head = newNode;
        }else{
            // 将新建节点的 next 指向旧节点
            newNode.next = current;
            // 将旧节点前驱的 next 指向新建的节点
            previous.next = newNode;
            // 设置被插入节点的 prev
            newNode.prev = previous;
        }
        this._size++;
        return newNode;
    }
    insertAt(newItem:T,pos:number):IDoubleNode<T>{
        // 创建新节点
        const newNode = new IDoubleNode(newItem);
        // 查找旧节点及其前驱节点
        const {current,previous} = this.findAt(pos);
        if(!current) return null;
        // 当 previous 为 null 时,说明是头节点
        if(!previous){
            newNode.next = current;
            // 设置当前节点的 prev
            current.prev = newNode;
            this._head = newNode;
        }else{
            // 将新建节点的 next 指向旧节点
            newNode.next = current;
            // 将旧节点前驱的 next 指向新建的节点
            previous.next = newNode;
            // 设置被插入节点的 prev
            newNode.prev = previous;
        }
        this._size++;
        return newNode;
    }
    toString():IDoubleNode<T>[]{
        const tmp:IDoubleNode<T>[] = [];
        let current:IDoubleNode<T> = this._head;
        while(current){
            tmp.push(current);
            current = current.next;
        }
        return tmp;
    }
}

循环链表

循环链表又分为单向循环链表和双向循环链表,分别是循环链表在单向链表和双向链表上的实现。
普通的单向链表的尾节点地址区域 next 指向的是 null,而循环单向链表的尾节点的地址区域指向的是头节点。如下图所示:

循环单向链表

普通双向链表的头节点地址区域 prev 指向的是 null,尾节点的地址区域的 next 指向的也是 null,在循环双向链表中,头节点的地址区域 prev 指向的是尾节点,而尾节点的地址区域 next 指向的是头节点。如下图所示:

循环双向链表

“循环”二字的意义就在于此。
在实现循环链表时,需要根据是循环单向链表还是循环双向链表对头节点或尾节点的地址区域进行相应的指向处理。我这里也不想实现了,介绍下基本概念吧,就酱~

完。

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

推荐阅读更多精彩内容

  • 本文内容:1、 什么是链表?2、 链表共分几类?3、 链表的 C 实现! 总表:《数据结构?》 工程代码 Gith...
    半纸渊阅读 39,550评论 0 54
  • 1 序 2016年6月25日夜,帝都,天下着大雨,拖着行李箱和同学在校门口照了最后一张合照,搬离寝室打车去了提前租...
    RichardJieChen阅读 5,016评论 0 12
  • 链表 概念 说到链表,coder们都不会陌生,在日常开发中或多或少都会用到它。它是链式存储的线性表,简称链表。链表...
    扈扈哈嘿阅读 2,045评论 0 5
  • 在我之前的文章曾经提到,因为《奇葩说》这个节目,我就被马东圈粉了,此后,凡是他的节目,我一定会第一时间收看。 幸运...
    小万管家阅读 269评论 0 0
  • 杰森·伯恩和他的原装人马, 今年暑期重磅回归。 《谍影重重5》中国票房两天超过1.05亿,周日破3亿,超过该系列第...
    辣八辣八阅读 171评论 0 0