数据结构基础--链表

目录

  • 基本性质
  • 链表的分类
    • 按连接方向分类
    • 按照有无循环分类
  • 链表问题代码实现的关键点
  • 链表插入和删除的注意事项
  • 链表翻转
  • 向一个有序的环境链表中插入一个节点,并保持依旧有序
  • 对于一个单链表,在不给定head的情况下删除指定node。要求时间复杂度O(1)
  • 给定一个链表,与一个数组num。要求实现荷兰国旗
  • 给定两个有序链表的head,打印共同部分
  • 给定一个单链表的head,实现一个调整链表的函数,使得每K个节点之间逆序,如果最后不足K个,则不调整
  • 判断一个链表是否为回文结构
  • 判断一个单链表是否有环,如有则返回入环节点。时间复杂度O(N),额外空间复杂度O(1)
  • 两个无环单链表是否相交,时间复杂度O(N+M),额外空间复杂度O(1)
  • 判断两个有环单链表是否相交,时间复杂度O(N+M),额外空间复杂度O(1)
  • 判断两个链表是否相交,并返回第一个相交的节点

基本性质

  • 链表问题算法难度不高,但考察代码的实现能力
  • 链表和数组一样,都是一种线性结构
  1. 数组是物理地址上一段连续的存储空间。
    可以通过下标直接获取元素
    当内容超出容量时需要重新定义数组。
  2. 链表空间不一定保持联系,为临时分配的。
    只能从链表的头部开始一个一个查找
    增删的效率高于数组,因为不需要更改内存结构

链表的分类

  • 按连接方向分类
  1. 单链表
    每个节点只能通过next指针,指向下一个节点。
  2. 双链表
    除了next指针之外,还有一个prev指针指向其上一个节点。
  • 按照有无循环分类
  1. 普通链表
    头无prev,尾无next。
  1. 循环链表
    首尾相接的链表。
    最后一个节点的next指针指向其第一个节点
    对于双链表,其第一个节点的prev指针指向最后一个节点。



链表问题代码实现的关键点

  • 链表调整函数的返回值,往往是节点类型

链表在调整过程中往往遇到改变头部的情况,如果头节点被改变则需要返回一个新头部。

  • 在调整链表的过程中,先采用画图的方式理清逻辑

注意那些指针变化了,同时注意对前后节点的影响。

  • 边界条件的处理

头节点,尾节点,空节点的特殊处理。


链表插入和删除的注意事项

  • 特殊处理链表为空或长度为1
  • 插入过程的调整

取得前后节点,将前节点的next指向新节点,新节点的next指向后节点。

  • 删除过程的调整

取得前后节点,将前一个节点的next指针指向后一个节点。

  • 头尾插入或删除

在逻辑的设计上应该考虑空节点的情况


链表翻转

  • 特殊处理链表为空或长度为1
  • 单链表的翻转

已翻转的头节点head,下一个节点now

  1. 将now节点的next指向head
  2. 将now节点设置为已翻转部分的新head

需要注意在执行1,2步骤之前需要一个变量来储存原now节点的next节点。
步骤2设置了新的head之后,将该节点作为新的now,继续翻转。


向一个有序的环境链表中插入一个节点,并保持依旧有序。

要求时间复杂度O(N),额外空间复杂度O(1)。

  • 如果链表为空

让新节点node自己成为环形链表,并返回node即可。

  • 如果链表不为空

令变量prve设为头节点,current设为第二个节点,两个节点同步移动。

  • 当有node<=prve && node>=current,则说明node应该插入二者之间

  • 若prve回到head但依旧没有合适的位置插入
    说明node为最大值或最小值,插入head之前即可。
    需要区分为两种情况下是否出现新的head,并返回。


对于一个单链表,在不给定head的情况下删除指定node。要求时间复杂度O(1)

  • 如果node.next不为空,也就是node不是尾节点

如果工程允许,可以将node.next的内容copy到node节点上,变相的删除了node节点的数据。

  • 如果node是尾节点

无法删除


给定一个链表,与一个数组num。要求实现荷兰国旗

  • 将链表遍历成数组,然后进行荷兰国旗排序,最后还原成链表。
  • 遍历链表的过程中使用三个小链表。小于,等于,大于。最后将三个链表串联。

给定两个有序链表的head,打印共同部分

  • 有一个为空直接返回
  • 采用外排的方式,直到有一个为空则停止。

给定一个单链表的head,实现一个调整链表的函数,使得每K个节点之间逆序,如果最后不足K个,则不调整。

  1. 链表为空,长度<k或者k<2
    直接返回
  • 通过栈结构,实现逆序
  1. 需要保留上次逆序的最后一位元素,修改其next。
  2. 最后段不足k个,直接不修改。值将上次逆序的最后一个元素next设置好。
  3. 第一组的第一个节点为头节点。
  • 不使用栈结构,手动逆序

判断一个链表是否为回文结构

  1. 将链表节点依次入栈,在弹出时与原链表依次比对。

  2. 使用快行指针,通过二倍速的方式遍历。依次将慢指针的节点压入栈中,当快节点遍历到末尾时,慢指针正好处于中间位置。
    继续移动慢指针,并与栈中弹出的元素做对比。(需要注意总量的奇偶)


  3. 将后半部分链表进行逆序处理,从两端同时进行遍历比对



判断一个单链表是否有环,如有则返回入环节点。时间复杂度O(N),额外空间复杂度O(1)

  • 1. 如果链表有结尾,则说明无环
  • 2.1 如果不要求额外空间复杂度,可以直接用哈希表比对。
  • 2.2 使用快行指针的方式

如果两指针相遇则表示有环,此时将快指针改为1,并从head重新同步移动,相遇处即为入环位置或者还有另一个证明

  • 代码
/// 获取入环节点
///
/// - Parameter node: 头节点
/// - Returns: 有环则返回入环节点,否则返回空
func getLoopNode(head : Node) -> Node {
    //链表长度为 0,1,2 不可能成环
    if head==nil || head.next==nil || head.next.next==nil {
        return nil
    }
    
    var slowP = head //慢行指针
    var fastP = head //快行指针
    
    while slowP != fastP {
        if slowP.next==nil || fastP.next.next==nil {  //链表有结尾,不可能成环
            return nil
        }
        slowP = slowP.next
        fastP = fastP.next.next
    }//运行到这里说明两指针相遇了
    
    
    //从head开始遍历再次相交则为入环点
    fastP = head
    while fastP != slowP {
        slowP = slowP.next
        fastP = fastP.next
    }
    
    return fastP
}

两个无环单链表是否相交,时间复杂度O(N+M),额外空间复杂度O(1)

  • 1. 先遍历两个链表确定长度
  • 2. 若两个链表结尾不同,则不相交
  • 3. 另长链表从短链表开始位置与短链表再次同步遍历,查看是否相同。
  • 代码
/// 两个无环单链表是否相交
///
/// - Parameters:
///   - head1: 链表1
///   - head2: 链表2
/// - Returns: 相交点或者为空
func noLoop(head1:Node ,head2:Node) -> Node {
    
    if head1==nil || head2==nil {
        return nil
    }
    var p1 = head1
    var p2 = head2
    
    //获取两个链表长度差值
    var n = 0
    while p1.next != nil {
        p1 = p1.next
        n+=1
    }
    while p2.next != nil {
        p2 = p2.next
        n-=1
    }
    
    if p1 != p2 { //若两个链表结尾不同,则一定不相交
        return nil
    }
    
    p1 = n>=0 ? head1:head2 //使 p1 指向较长的链表
    p2 = p2==head1 ? head2:head1 //使p2 指向另一个链表
    
    n = abs(n) //取绝对值
    while n>0 {//将长链表移动n次。
        p1 = p1.next
        n-=1
    }
    
    //查找链表上第一个相同的点
    while p1 != p2 {
        p1 = p1.next
        p2 = p2.next
    }
    
    return p1
}

判断两个有环单链表是否相交,时间复杂度O(N+M),额外空间复杂度O(1)

首先都需要先去定单独的入环节点,然后

  1. 是否入环之前已经相交


  2. 是否入环时才相交,则入环位置节点相同



    如果相交点为loop1或者loop2,则为入环时才相交

  3. 入环后才相交
    循环其中一个环,若遇到另一个的入环节点则返回。


  4. 否则,两链表并未相交

  • 代码
/// 两个有环单链表是否相交
///
/// - Parameters:
///   - head1: 链表1
///   - head2: 链表2
///   - loop1: 链表1入环点
///   - loop2: 链表2入环点
/// - Returns: 相交点或者为空
func bothLoop(head1:Node ,head2:Node ,loop1:Node ,loop2:Node) -> Node {
    
    if head1==nil || head2==nil{
        return nil
    }
    
    if loop1 == loop2 { //两个链表在入环之前已经相交
        var p1 = head1
        var p2 = head2
        
        //获取两个链表长度差值
        var n = 0
        while p1.next != loop1 {
            p1 = p1.next
            n+=1
        }
        while p2.next != loop2 {
            p2 = p2.next
            n-=1
        }
        
        p1 = n>=0 ? head1:head2 //使 p1 指向较长的链表
        p2 = p2==head1 ? head2:head1 //使p2 指向另一个链表
        
        n = abs(n) //取绝对值
        while n>0 {//将长链表移动n次。
            p1 = p1.next
            n-=1
        }
        
        //查找链表上第一个相同的点
        while p1 != p2 {
            p1 = p1.next
            p2 = p2.next
        }
        
        return p1
    }else { //两个链表在入环之后才相交
        var p1 = loop1.next
        var p2 = loop2
        while p1 == loop1 { //循环loop1一次
            p1 = p1.next
            if p1 == p2 {
                return p1
            }
        }
        return nil  //并未相交
    }
}


判断两个链表是否相交,并返回第一个相交的节点

  1. 尝试找到各自的入环节点

  2. 若一个有环一个无环,则不相交

  3. 若都为无环,则按照上文《两个无环单链表是否相交》的方式查找

  4. 若都为有环,则按照上文《判断两个有环单链表是否相交》的方式查找

  • 代码
func getIntersectNode(head1:Node,head2:Node) -> Node {
    if head1 == nil || head2 == nil {
        return nil
    }
    
    //获取两个入环节点
    var loop1 = getLoopNode(head: head1)
    var loop2 = getLoopNode(head: head2)
    
    if (loop1 == nil && loop2 != nil)||(loop1 != nil && loop2==nil) {
        return nil //一个有环一个无环,肯定不相交
    }
    
    if loop1==nil || loop2==nil {//两个链表都不为环型结构
        return noLoop(head1: head1, head2: head2)
    }else { //两个环形链表
        return bothLoop(head1: head1, head2: head2, loop1: loop1, loop2: loop2)
    }
}


参考资料

左神牛课网算法课

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