重排链表

此题为抖音现场面试的算法题,比较经典。

解题思路

给定链表 1->2->3->4->5, 重新排列为 1->5->2->4->3.

通过观察,可以将重排链表分解为以下三个步骤:

  1. 首先重新排列后,链表的中心节点会变为最后一个节点。所以需要先找到链表的中心节点:876. 链表的中间结点
  2. 可以按照中心节点将原始链表划分为左右两个链表。
    2.1. 按照中心节点将原始链表划分为左右两个链表,左链表:1->2->3 右链表:4->5。
    2.2. 将右链表反转,就正好是重排链表交换的顺序,右链表反转:5->4。反转链表:206. 反转链表
  3. 合并两个链表,将右链表插入到左链表,即可重新排列成:1->5->2->4->3.

代码

class Solution {
    public void reorderList(ListNode head) {
        if(head == null){
            return ;
        }
        //1. 使用快慢指针,找出链表的中心节点。
        // 1->2->3->4->5,中心节点为3
        ListNode middle = middleNode(head);

        //2. 将原始链表按照中心链表分割为两个链表,并将右链表反转
        //2.1 原始链表:1->2->3->4->5 左链表:1->2->3 右链表:4->5
        ListNode left = head;
        ListNode right = middle.next;
        middle.next = null;

        //2.2 反转右链表
        //原始右链表:4->5 反转后:5->4
        right = reverse(right);

        //3. 合并两个链表,将右链表插入到左链表
        //左链表:1->2->3 右链表:4->5 合并后:1->5->2->4->3
        merge(left,right);
    }

    //1. 使用快慢指针,找出链表的中心节点
    public ListNode middleNode(ListNode head) {
        ListNode slow = head;
        ListNode fast = head;

        while (fast != null && fast.next != null) {
            slow = slow.next;
            fast = fast.next.next;
        }
        return slow;
    }

    //2. 通过递归反转链表
    public ListNode reverse(ListNode head) {
        if (head == null || head.next == null) {
            return head;
        }
        ListNode last = reverse(head.next);
        head.next.next = head;
        head.next = null;
        return last;
    }

    //3. 合并两个链表,将右链表插入到左链表
    public void merge(ListNode left, ListNode right){
        ListNode leftTemp;
        ListNode rightTemp;
        while (left.next != null && right!= null) {
            //1. 保存next节点
            leftTemp = left.next;
            rightTemp = right.next;

            //2. 将右链表的第一个节点插入到左链表中
            // 左链表:1->2->3 右链表:5->4 
            // 合并后的左链表:1->5->2->3 
            left.next = right;
            right.next = leftTemp;

            //3. 移动left和right指针
            //左链表变为:2->3 右链表变为:4
            left = leftTemp;
            right = rightTemp;
        }
    }
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值