将单向链表按某值划分成左边小、中间相等、右边大的形式

将单向链表按某值划分成左边小、中间相等、右边大的形式

【题目】 给定一个单向链表的头节点head,节点的值类型是整型,再给定一个 整 数pivot。实现一个调整链表的函数,将链表调整为左部分都是值小于 pivot 的节点,中间部分都是值等于pivot的节点,右部分都是值大于 pivot的节点。 除这个要求外,对调整后的节点顺序没有更多的要求。 例如:链表9->0->4->5- >1,pivot=3。 调整后链表可以是1->0->4->9->5,也可以是0->1->9->5->4。总 之,满 足左部分都是小于3的节点,中间部分都是等于3的节点(本例中这个部 分为空),右部分都是大于3的节点即可。对某部分内部的节点顺序不做 要求。

进阶1: 在原问题的要求之上再增加如下两个要求。 在左、中、右三个部分的内部也做顺序要求,要求每部分里的节点从左 到右的 顺序与原链表中节点的先后次序一致。 例如:链表9->0->4->5->1,pivot=3。 调整后的链表是0->1->9->4->5。 在满足原问题要求的同时,左部分节点从左到 右为0、1。在原链表中也 是先出现0,后出现1;中间部分在本例中为空,不再 讨论;右部分节点 从左到右为9、4、5。在原链表中也是先出现9,然后出现4, 最后出现5。

进阶2:如果链表长度为N,时间复杂度请达到O(N),额外空间复杂度请达到O(1)

思路:普通方法 用数组存储链表中的数据,排序,再将这些数据用链表串起来
进阶:三个指针,分别指向大于,等于,小于,最后把他们串起来

void insert_back(Node*&obj, Node*&add){
	if (obj == NULL){
		add->_next = NULL;
		obj = add;
	}
		
	else{
		Node*temp = obj;
		while (temp->_next != NULL){
			temp = temp->_next;
		}
		add->_next = NULL;
		temp->_next = add;
	}
}
void insert_list(Node*&obj, Node*&add){
	if (obj == NULL){
		obj = add;
	}
		
	else{
		Node*temp = obj;
		while (temp->_next != NULL){
			temp = temp->_next;
		}
		temp->_next = add;
	}
}
void sort_list(Node*&head,int num){
	Node*small, *big, *eql,*cur,*temp;
	small = NULL;
	big = NULL;
	eql = NULL;
	cur = head;
	while (cur){
		temp = cur;
		cur = cur->_next;
		if (temp->data < num){
			insert_back(small, temp);
		}
		else if (temp->data>num)
			insert_back(big, temp);
		else
			insert_back(eql, temp);
	}
	insert_list(small, eql);
	insert_list(eql, big);
	head = small;
}

总结:代码本身难度不大,但是细节容易出错,注意尾插一个节点,和连接两个链表时的区别(不要忘记空链表时的特殊处理)。

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值