算法题,c++,合并K个升序链表,原链表修改链接顺序降低空间复杂度,分治合并降低时间复杂度

算法题,c++,合并K个升序链表,原链表修改链接顺序降低空间复杂度,分治合并降低时间复杂度

题目:leetcode
给你一个链表数组,每个链表都已经按升序排列。请你将所有链表合并到一个升序链表中,返回合并后的链表。

链表结点结构:

 typedef struct ListNode {
      int val;
      ListNode *next;
      ListNode() : val(0), next(nullptr) {}
      ListNode(int x) : val(x), next(nullptr) {}
      ListNode(int x, ListNode *next) : val(x), next(next) {}
 }ListNode;

思路:
1、K个链表数据量很大,不适合新建容器或新建链表然后排序,可以利用链表本身的next属性,改变每个结点的后接结点,以达到节省空间的效果。
2、K个链表同时对首结点的数据进行比较不方便,如果是两个链表就可以很方便的使用if进行判断,因此在K个链表中两两相合,两两相合的结果再次两两相合,合到只有一个链表,就是最终的结果了。

代码与每一行的注释,一定注意看注释,看注释,不然有可能看不懂代码的:

//合并两个链表为一个链表
ListNode* AddToOne(ListNode* AList, ListNode* BList)
{
	//判断是否传入的两个链表中有空的链表
	if (!AList || !BList)
	{
		return (AList ? AList : BList);
	}

	//用于组合新链表的链表头
	ListNode* Tail;

	//链表头不能直接使用,新建一个临时结点提供给链表头
	//这个空链表头还用于找到链表的头,避免在合并完以后无法获取到链表的起点
	//临时结点还用于返回链表头,通过return Head.next
	ListNode Head;
	Tail = &Head;

	//用于遍历两个链表的所有内容和比较链表结点数据的指针
	ListNode* pA = AList;
	ListNode* pB = BList;

	//当两个遍历的结点A,B都有值时,说明两个链表中的数据还没有遍历完,
	//只要有一个遍历完了,就表示两个合并为一个了,且无法继续进行比较了
	//因此while循环的判断条件是两个结点都有值而不是空
	while (pA && pB)
	{
		//哪个小就把哪个放到新链表的尾部
		if (pA->val < pB->val)
		{
			Tail->next = pA;
			pA = pA->next;
		}
		else
		{
			Tail->next = pB;
			pB = pB->next;
		}
		//装入尾部后将新链表的当前结点向后移,保持在最后位
		Tail = Tail->next;
	}

	//判断结束后,一个链表变成空了,剩下还有一部分链表需要另外加入到新链表
	Tail->next = (pA ? pA : pB);

	return Head.next;
}

//使用分治算法将链表vector进行分治合并

//每一次合并只能合并两个下标的链表,将两个链表的所在下标表示为left,right
ListNode* merge(vector<ListNode*>& Data,int left,int right)
{
	//当左下标和右下标相等时,返回该下标对应的链表就可以
	if (left == right)
	{
		return Data[left];
	}

	//当左下标和右下标紧跟着时,就可以合并了,然后将合并的结果返回
	else if(left + 1 == right)
	{
		return AddToOne(Data[left], Data[right]);
	}

	//当左下标和右下标没有相等,没有紧跟着时,递归下去
	//向下递归的规律为:从最左边到中间的合并,中间后一个结点到最右边的合并,得到的两个链表合并后将链表返回
	else
	{
		return AddToOne(merge(Data, left, (left + right) / 2 ), merge(Data, (left + right) / 2 + 1, right));
	}
}

ListNode* mergeKLists(vector<ListNode*>& Data)
{
	//传入的vector有可能是个空的,为了避免后面的size-1为负数,有必要对size的值进行判断
	if (Data.size() == 0)
	{
		return NULL;
	}
	return merge(Data, 0, Data.size() - 1);
}
上一篇:【2022初春】【LeetCode】83. 删除排序链表中的重复元素


下一篇:链表--19.删除链表的倒数第N个结点