《牛客每日一题》链表分割、输出链表的倒数第k个结点

✅作者简介:大家好,我是小鱼儿,大家可以叫我鱼儿

📒博客首页:是小鱼儿哈

🔥系列专栏:一起刷好题

🌻每日一句:努力不是重点,常态化才是关键。真正努力的人,能随时进入任何角色,在过程中找到感觉和快乐。

💖博主也在学习阶段,如发现问题请告知,非常感谢💖

目录

一、输出链表的倒数第k个结点 

二、链表分割 


一、输出链表的倒数第k个结点 

 题目链接:输出链表的倒数第k个结点

 

📝思路一:通过遍历链表,求得链表的长度,然后求得返回的是正数的第一个结点

public class Solution {
    public ListNode FindKthToTail(ListNode head,int k) {
        ListNode cur = head;
        int count = 0;
        while (cur != null) {
            cur = cur.next; count++;  // 通过遍历求出链表的结点数目
        }
        if (k > count) return null;   // 当k > 链表的长度时,直接返回null
        if (head == null || k <= 0) return null; // 处理特殊情况
        ListNode tmp = head;
        // 倒数第k个结点,就是相当于是正数第count - k个结点
        for (int i = 0; i < count - k; ++i) {
            tmp = tmp.next;
        }
        return tmp;  // 相当于是把倒数第k个结点之后的那一串结点给返回了
    }
}

 

 

📝 思路二、不求长度,用双指针来做

即快指针fast先走k步,然后慢指针slow再开始走,当fast走到链表的结尾时,slow刚好走到倒数第k个结点处(代码里有详细解释)

public class Solution {
    public ListNode FindKthToTail(ListNode head,int k) {
        if (head == null || k <= 0) return null; // 处理特殊情况
        // 定义两个快慢指针
        ListNode fast = head, slow = head;
        for (int i = 0; i < k; ++i) { // 即fast先走k步
            if (fast == null){  
                return null;
            } 
            else {
                fast = fast.next;
            } 
        }
        // 之后fast和slow再一起走,当fast走到链表的结尾处时,slow刚好走的链表第k个结点处
        // 为什么呢?因为比方倒数第k个结点是正数第m个结点,那么m + k == 链表的长度,
        // fast先走了k个结点,那么fast再走m个结点不就正好到链表结尾了吗?此时slow不就刚好走了k个结点吗!
        while (fast != null) {
            slow = slow.next;
            fast = fast.next;
        }
        return slow; // 返回该结点
    }
}

🤔有小伙伴可能会问了for循环里的判断是什么意思呢?

😂主要是处理 k > 链表的长度这种情况 

当循环这个位置(If 语句里)出现fast等于null的情况,说明一定是k > 链表的长度

因为if语句判断fast是否为空我放在了前面,即使k == 链表的长度,此时在这个位置fast也不可能为空,最多经过了下面的fast = fast.next此时才为空,但同时这个时候循环马上就要退出了

所以当在if(里出现fast 为空的情况时候,一定是 K > 链表的长度


 

二、链表分割 

题目链接 

链表分割——牛客

 

 

📝思路:设置两个链表头,遍历原链表,一个追加小数链表,一个追加大数链表,最后将小数链表粘到大数链表前边即为结果。

public class Partition {
    public ListNode partition(ListNode pHead, int x) {
        // write code here
        if (pHead == null) return pHead;
        // 为了便于操作我们对要用到的小数链表(存放大于x值的链表)和大数链表分别定义他们各自的虚拟头结点
        ListNode dummyHead1 = new ListNode(-1), dummyHead2 = new ListNode(-1);
        ListNode cur1 = dummyHead1, cur2 = dummyHead2; // 为防止链表丢失,我们不能直接使用虚拟头节点,
        ListNode curp = pHead; // curp指向原链表,我们接下来也是用curp来对原链表进行遍历操作的
        while (curp != null) {
            if (curp.val < x) {
                cur1.next = curp; // 当curp.val小于x时,给小数链表添加元素
                cur1 = cur1.next;
            }
            else {
                cur2.next = curp; // 否则给大数链表增添元素
                cur2 = cur2.next;
            }
            curp = curp.next;  // 更新curp对原来链表的指向,完成遍历操作
        }
        // 此时的cur1指向小数链表的最后一个结点,dummyHead2.next是大数链表的第一个结点
        // 我们要把他们两个链接起来
        cur1.next = dummyHead2.next; 
        cur2.next = null; // cur2是大数链表的最后一个元素,指向空就行了
        return dummyHead1.next; // 返回小数链表的第一个元素
    }
}

  • 5
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 2
    评论
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

是小鱼儿哈

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值