先来一道看起来简单的题目:
题目1:将两个升序链表合并为一个新的 升序 链表并返回。新链表是通过拼接给定的两个链表的所有节点组成的。
输入:l1 = [1,2,4], l2 = [1,3,4] 输出:[1,1,2,3,4,4]
解题思路:我们可以把两个链表合并进入一个新链表中;也可以寻找两个链表的头结点比较小的值,将另一个链表(L2)的节点插入到头结点比较小的链表(L1)中
package code.code_02;
/**
* 力扣原题 https://leetcode.cn/problems/merge-two-sorted-lists/
* 将两个升序链表合并为一个新的 升序 链表并返回。新链表是通过拼接给定的两个链表的所有节点组成的。
* 输入:l1 = [1,2,4], l2 = [1,3,4]
* 输出:[1,1,2,3,4,4]
*
* 解题思路: 此题比较简单,就是单纯的合并2个链表,将数据合并进入长链表中
* 先检查哪个链表的首节点比较小,数据往首节点比较小的链表中插入。
*
* 一开始我是准备往长链表中插入数据,但是发现有可能频繁更新长链表的头结点,逻辑会比较复杂,这不是一个好思路
*/
public class MergeTwoListTest {
public class ListNode {
int val;
ListNode next;
ListNode() {}
ListNode(int val) { this.val = val; }
ListNode(int val, ListNode next) { this.val = val; this.next = next; }
}
private ListNode getInsertNode (ListNode minStart, int value)
{
ListNode node = minStart;
while (minStart != null && minStart.val < value) {
if (minStart.next == null || minStart.next.val >= value) {
node = minStart;
}
minStart = minStart.next;
}
return node;
}
public ListNode mergeTwoLists(ListNode list1, ListNode list2)
{
if (list1 == null) {
return list2;
}
else if (list2 == null) {
return list1;
}
ListNode minStart = (list1.val <= list2.val) ? list1 : list2;
ListNode maxStart = (minStart == list1) ? list2 : list1;
ListNode nextMin = null;
ListNode nextMax = null;
ListNode node = minStart;
while (maxStart != null) {
ListNode InserAfterNode = this.getInsertNode(minStart, maxStart.val);
//如果要插入的链表已经到达链表尾部,
nextMin = InserAfterNode.next;
nextMax = maxStart.next;
maxStart.next = null;
//插入节点
InserAfterNode.next = maxStart;
InserAfterNode = maxStart;
if (nextMin != null) {
//将新插入的节点,持有原最小节点的下一个节点
InserAfterNode.next = nextMin;
/**
* 此处要尤为注意,需要考虑重复元素的问题,千万不要
* 这样:
* InserAfterNode = nextMin
* minStart = InserAfterNode
*
* 如果按照以上的写法,会出错,case为
* l1 = [-10,-9,-6,-4,1,9,9]
* l2 = [-5,-3,0,7,8,8]
*/
minStart = InserAfterNode;
}
//待插入节点来到下一个节点位置
maxStart = nextMax;
}
return node;
}
// 先设计打印方法,方便检查逆转后结果
public void printNode (ListNode node)
{
if (node == null) {
System.out.println("链表不存在");
}
System.out.println("当前节点的值为: " + node.val);
//递归的方式逐层打印Node的子节点
if(node.next != null) {
printNode(node.next);
}
}
public static void main(String[] args) {
MergeTwoListTest test = new MergeTwoListTest();
//链表1
/* ListNode l1 = test.new ListNode(1);
l1.next = test.new ListNode(2);
l1.next.next = test.new ListNode(4);
//链表2
ListNode l2 = test.new ListNode(1);
l2.next = test.new ListNode(3);
l2.next.next = test.new ListNode(4);*/
ListNode l1 = test.new ListNode(-10);
l1.next = test.new ListNode(-9);
l1.next.next = test.new ListNode(-6);
l1.next.next.next = test.new ListNode(-4);
l1.next.next.next.next = test.new ListNode(1);
l1.next.next.next.next.next = test.new ListNode(9);
l1.next.next.next.next.next.next = test.new ListNode(9);
//链表2
ListNode l2 = test.new ListNode(-5);
l2.next = test.new ListNode(-3);
l2.next.next = test.new ListNode(0);
l2.next.next.next = test.new ListNode(7);
l2.next.next.next.next = test.new ListNode(8);
l2.next.next.next.next.next = test.new ListNode(8);
ListNode nodeList = test.mergeTwoLists(l1, l2);
test.printNode(nodeList);
}
}
实测结果:
题目2: 合并K个升序链表
输入:lists = [[1,4,5],[1,3,4],[2,6]] 输出:[1,1,2,3,4,4,5,6]
package code1.code_03;
import java.util.Comparator;
import java.util.PriorityQueue;
/**
* LeetCode: https://leetcode.cn/problems/merge-k-sorted-lists/
* 题目: 合并K个升序链表
* 输入:lists = [[1,4,5],[1,3,4],[2,6]]
* 输出:[1,1,2,3,4,4,5,6]
* 解释:链表数组如下:
* [
* 1->4->5,
* 1->3->4,
* 2->6
* ]
* 将它们合并到一个有序链表中得到。
* 1->1->2->3->4->4->5->6
*/
public class Code01_MergeKNodeList {
public class ListNode {
int val;
ListNode next;
ListNode() {}
ListNode(int val) { this.val = val; }
ListNode(int val, ListNode next) { this.val = val; this.next = next; }
}
public ListNode mergeKLists(ListNode[] lists) {
if (lists == null) {
return null;
}
if (lists.length == 1) {
return lists[0];
}
//此处也可以用ArrayList,但是ArrayList的排序需要调用sort方法,在下面的while会多次
//调用,结合性能因素,此处使用的是优先队列
PriorityQueue queue = new PriorityQueue<ListNode>(new Comparator<ListNode>() {
/**
* 任何比较器、
* 如果负数,第一个排前面
* 如果整数,第二个排前面
* 0无所谓
*/
@Override
public int compare(ListNode o1, ListNode o2) {
return o1.val - o2.val;
}
});
//需要遍历数组,并且把每个链表的头结点放入queue中
for (int i =0; i < lists.length; i++) {
if (lists[i] != null) {
queue.add(lists[i]);
}
}
//第一次拿到的链表的值就是最小值,我们将其他链表合并进入这个链表中
ListNode head = (ListNode) queue.poll();
//如果头节点还有next节点,那么我们需要将next节点丢入
//优先队列中进行再次排序
if (head != null && head.next != null) {
queue.add(head.next);
}
ListNode minValNode = null;
ListNode curNode = head;
while (!queue.isEmpty()) {
minValNode = (ListNode) queue.poll();
curNode.next = minValNode;
curNode = minValNode;
if (curNode.next != null) {
queue.add(curNode.next);
}
}
return head;
}
public static void printNode (ListNode node) {
if (node == null) {
System.out.println("链表不存在");
}
System.out.println("当前链表的值为: " + node.val);
//递归的方式逐层打印Node的子节点
if(node.next != null) {
printNode(node.next);
}
}
public static void main(String[] args) {
Code01_MergeKNodeList test = new Code01_MergeKNodeList();
ListNode node1 = test.new ListNode(2);
node1.next = test.new ListNode(4);
node1.next.next = test.new ListNode(5);
ListNode node2 = test.new ListNode(1);
node2.next = test.new ListNode(3);
node2.next.next = test.new ListNode(4);
ListNode node3 = test.new ListNode(2);
node3.next = test.new ListNode(6);
ListNode[] list = new ListNode[3];
list[0] = node1;
list[1] = node2;
list[2] = node3;
ListNode mergedNode = test.mergeKLists(list);
printNode(mergedNode);
}
}