zoukankan      html  css  js  c++  java
  • 【LeetCode148】Sort List★★bug

    1.题目描述:

    2.解题思路:

       本题是要堆一个链表进行排序,并且要求时间复杂度为 O(n log n)。很明显,要用到分治的思想,用二分法进行归并排序:找到链表的middle节点,然后递归对前半部分和后半部分分别进行归并排序,最后对两个已排好序的链表进行Merge。

      分为三步:

      (1)找到中间结点,将链表分割成两部分。这里用到快慢两个指针的方法。

      (2)对前后每一部分分别进行归并排序。这里用到递归。

      (3)对两个已排好序的链表进行合并。这里用到前面merge 2 list的方法。

      

    3.Java代码:

    //public class LeetCode148 为测试代码
    public class LeetCode148 {
        public static void main(String[] args) {
            ListNode n1=new ListNode(3),n2=new ListNode(2),n3=new ListNode(1),n4=new ListNode(5);
            ListNode p=n1;
            n1.next=n2;
            n2.next=n3;
            n3.next=n4;
            System.out.print("初始时链表:"+p.val);
            while(p.next!=null){
                System.out.print("->"+p.next.val);
                p=p.next;
            }
            System.out.println();
            ListNode head=new Solution().sortList(n1);
            System.out.print("排序后链表:"+head.val);
            while(head.next!=null){
                System.out.print("->"+head.next.val);
                head=head.next;
            }
            }
    }
    
    //class Solution为ac代码
    class Solution {
        public ListNode sortList(ListNode head) {
            if(head==null||head.next==null) return head;
            // step 1. 将链表分割为两部分
            ListNode pre=null,slow=head,fast=head;
            while(fast!=null&&fast.next!=null){
                pre=slow;//pre指针是为了记住slow的前一个节点位置
                slow=slow.next;
                fast=fast.next.next;
            }
            pre.next=null;//将链表从pre和slow之间断开
            // step 2. 对每一部分进行排序
            ListNode l1=sortList(head);
            ListNode l2=sortList(slow);
            // step 3. merge l1 and l2
            return merge2Lists(l1,l2);
            }
    
        private ListNode merge2Lists(ListNode l1, ListNode l2) {
           ListNode fakeHead=new ListNode(0);
           ListNode p=fakeHead;
           while(l1!=null&&l2!=null){
               if(l1.val<l2.val){
                   p.next=l1;
                   l1=l1.next;
               }else{
                   p.next=l2;
                   l2=l2.next;
               }
               p=p.next;
               }
           if(l1!=null) p.next=l1;
           if(l2!=null) p.next=l2;
           return fakeHead.next;
           }
    }
    class ListNode{
        int val;
        ListNode next;
        ListNode(int x) { val = x; }
    }

    测试结果:

    思考:我将merge2Lists方法写成递归的写法时,即

     1 private ListNode merge2Lists(ListNode l1, ListNode l2) {
     2        if(l1==null) return l2;
     3        if(l2==null) return l1;
     4        if(l1.val<l2.val){
     5            l1.next=merge2Lists(l1.next, l2);
     6            return l1;
     7        }else{
     8            l2.next=merge2Lists(l1, l2.next);
     9            return l2;
    10        }
    11        }

    这时在eclipse上测试完全没问题,但是在LeetCode上提交不通过,至今没想明白说明问题,如果哪位大神明白为什么,一定告诉我。。。

  • 相关阅读:
    字符串:序列自动机
    图论学习——最大团与最大独立集
    点分治
    图论:Johnson全源最短路
    停止更新博客
    将Eclipse中现有的java类生成类图
    problem:SVN error: (501 Not Implemented)
    SVN 修改URL路径
    eclipse中,把java函数代码折叠/展开
    Build类
  • 原文地址:https://www.cnblogs.com/zhangboy/p/6497693.html
Copyright © 2011-2022 走看看