解题思路: 1.用输入的数据创建好两个链表。

2.将两个无序链表合并成一个链表:用一个变量记录链表1的尾结点,使其下个节点指向链表2。

3. 对合并后的链表进行排序:排序采取排序算法中的归并排序。个人认为对于链表排序实现采取递归方式较为容易。


注意事项:1. 创建好结点类,成员变量为学号,成绩,和用于指向下一个节点的变量。

              2. 注意结点指向问题,局域变量作用范围

              3. 递归的退出条件


参考代码:

import java.util.Scanner;

public class Main {
   public static void main(String[] args) {
       Scanner sc = new Scanner(System.in);
       int i1 = sc.nextInt();
       int i2 = sc.nextInt();
       ListNode list1 = new ListNode(sc.nextInt(),sc.nextInt());
       ListNode head1 = list1;
       for (int i = 0; i < i1-1; i++) {
           head1.next = new ListNode(sc.nextInt(),sc.nextInt());

// 确保head1为链表1的尾节点
           if (i!=i1-1) {
               head1 = head1.next;
           }
       }

       ListNode list2 = new ListNode(sc.nextInt(),sc.nextInt());
       ListNode head2 = list2;
       for (int i = 0; i < i2-1; i++) {
           head2.next = new ListNode(sc.nextInt(), sc.nextInt());

// head2为链表2的尾节点
           if (i != i2 - 1) {
               head2 = head2.next;
           }
       }

// 链表1的尾节点的下一节点指向链表2
       head1.next = list2;


//        上面为合并两个链表为一个无序链表,下面对其进行归并排序
       list1 = merge(list1);

// 循环输出节点属性
       while (list1!=null){
           System.out.println(list1.id+" "+ list1.grade);
           list1 =list1.next;
       }
   }


// 归并排序中的合并排序操作(相当于对两个有序链表合并成一个有序链表)
   public static ListNode sort(ListNode l1,ListNode l2){

// 当两个链表中其中一个链表为空则返回另一个链表(递归退出条件)
       if (l1==null){
           return l2;
       }
       if (l2==null){
           return l1;
       }

// 学号小的结点指向学号大的结点
       if (l1.id<l2.id){

    // 将 l1(学号小的头结点)的下一个结点作为下一个递归中的头结点 l1, l2 不改变
           l1.next = sort(l1.next,l2);

    // 将当前分链表的头节点返回
           return l1;
       }else {
            l2.next = sort(l1,l2.next);
           return l2;
       }
   }


// 归并排序中的拆分操作
   public static ListNode merge(ListNode head){

// 头节点为空或下一个结点为空返回头节点(递归退出条件)
       if (head==null || head.next==null){
           return head;
       }

// 快慢指针 (用于寻找链表的中间结点,即移动完后慢指针slow是中间结点)

       ListNode slow = head;            
       ListNode fast =head                  
// 当fast的下一个结点为空(链表结点个数为偶数情况)或fast的下下个结点为空(链表结点个数为奇数情况)                

       while (true){
           if (fast.next==null || fast.next.next==null){    
               break;
           }                                  

// 慢指针每次移动一次 快指针每次移动两次,可以较为均匀的分成两个链表                              
           slow = slow.next
           fast = fast.next.next
       }                                                                                                                                                                            

// head 到slow 为 l1 链表,slow.next到fast为 l2 链表           
       ListNode l1 = head
       ListNode l2 = slow.next   

// 拆分成两个链表,将 l1 链表的尾节点slow的下一个节点指向null

       slow.next =null;

//继续拆分直到为单个节点为止
       l1 = merge(l1);
       l2 = merge(l2);

// 返回合并排序过后的链表头节点
       return sort(l1,l2);
   }
}


//结点类

class ListNode {
   int id;
   int grade;

//用于指向下一个节点
   ListNode next;
   ListNode() {}
   ListNode(int val,int grade) { this.id = val; this.grade=grade;}
   ListNode(int val, ListNode next) { this.id = val; this.next = next; }
}

点赞(0)
 

0.0分

3 人评分

C语言网提供由在职研发工程师或ACM蓝桥杯竞赛优秀选手录制的视频教程,并配有习题和答疑,点击了解:

一点编程也不会写的:零基础C语言学练课程

解决困扰你多年的C语言疑难杂症特性的C语言进阶课程

从零到写出一个爬虫的Python编程课程

只会语法写不出代码?手把手带你写100个编程真题的编程百练课程

信息学奥赛或C++选手的 必学C++课程

蓝桥杯ACM、信息学奥赛的必学课程:算法竞赛课入门课程

手把手讲解近五年真题的蓝桥杯辅导课程

评论列表 共有 0 条评论

暂无评论