链表的归并排序,无需辅助外存

rt, 最近的面试题,面的时候敲的血崩,特此写下博客以作留念

如有错误,欢迎指正

class Node{

    int val;

    Node next;

}

import java.util.Random;

class Node {
    public int val;
    public Node next;
    public Node(int val) {
        this.val = val;
        this.next = null;
    }
}
public class Test {
    public static void main(String[] args) {
        Random random = new Random(47);
        Node head = new Node(10);
        Node p = head;
        for(int i = 0; i < 9; ++i) {
            p.next = new Node(random.nextInt(20));
            p = p.next;
        }
        System.out.print("归并前:");
        for (Node i = head; i != null; i = i.next)
            System.out.print(i.val + " ");
        System.out.println();
        head = MergeSort(head, 10);

        System.out.print("归并后:");
        for (Node i = head; i != null; i = i.next)
            System.out.print(i.val + " ");
        System.out.println();
    }

    //传入链表头和链表长度,返回新链表头,对以head开始的长度为len的链表排序
    private static Node MergeSort(Node head, int len) {
        if (len > 1) {
            int m = len / 2;
            head = MergeSort(head, m);
            Node p = head;
            for (int i = 0; i < m - 1; ++i)
                p = p.next;
            p.next = MergeSort(p.next, len - m);
            head = merge(head, len, m);
        }
        return head;
    }

    //对以head开始的长度为len的链表合并, [1, m]和[m+1, len]的两条链表
    private static Node merge(Node head, int len, int m) {
        Node p1 = head;
        Node p1Pre = null;
        Node p2 = head.next;
        Node p2Pre = head;
        for (int i = 0; i < m - 1; ++i) {
            p2 = p2.next;
            p2Pre = p2Pre.next;
        }
        int l = 1;
        int r = m + 1;
        while (l <= m && r <= len) {        //将链表归并到p1链上
            if (p1.val <= p2.val) {
                p1Pre = p1;
                p1 = p1.next;
                ++l;
            } else {             //将p2指向的节点移到p1的前面
                p2Pre.next = p2.next;
                p2.next = p1;
                if (p1Pre == null) {
                    head = p2;
                    p1Pre = head;
                } else {
                    p1Pre.next = p2;
                    p1Pre = p2;
                }
                p2 = p2Pre.next;
                ++r;
            }
        }
        return head;
    }
}

虽然链表的归并无需辅存,但链表的查找增加了时间复杂度,而且写的时候容易出错,链表的归并也并不好用

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值