在链表上实现 Partition 以及荷兰国旗问题

作者:Grey

原文地址:

博客园:在链表上实现 Partition 以及荷兰国旗问题

CSDN:在链表上实现 Partition 以及荷兰国旗问题

题目描述

给你一个链表的头节点 head 和一个特定值 x ,请你对链表进行分隔,使得所有 小于 x 的节点都出现在 大于或等于 x 的节点之前。

你应当 保留 两个分区中每个节点的初始相对位置。

题目链接见:LeetCode 86. Partition List

主要思路

本题可以借鉴数组的 Partition 操作,参考:荷兰国旗问题与快速排序算法

Partition 操作就是荷兰国旗的一种特殊情况而已。

我们可以把本题的难度稍微升级一下:如何在链表上实现荷兰国旗问题?

第一种解法就是将链表先转换成数组,然后在数组上实现荷兰国旗问题,最后将数组还原为链表并返回,该方法的时间复杂度是O(N),空间复杂度是O(N),不是最优解。

第二种解法是用有限几个变量来实现,在同样O(N)的时间复杂度的情况下,空间复杂度可以做到O(1),设置如下几个变量

ListNode sH = null; // 小于区域的头结点
ListNode sT = null; // 小于区域的尾结点
ListNode eH = null; // 等于区域的头结点
ListNode eT = null; // 等于区域的尾结点
ListNode mH = null; // 大于区域的头结点
ListNode mT = null; // 大于区域的尾结点
ListNode next; // 记录遍历到的结点的下一个结点

接下来开始遍历链表,进行主流程处理,伪代码如下

while (head != null) {
next = head.next;
// 如果head.val < pivot,则通过sH,sT构造小于区域的链表
// 如果head.val == pivot,则通过eH,eT构造小于区域的链表
// 如果head.val > pivot,则通过mH,mT构造小于区域的链表
head = next;
}
// 把三个区域的链表串联起来即可。

构造每个区域的链表的时候,还要考虑链表是第一次被构造还是已经构造了,以小于区域的链表为例,如果是第一次构造,则sH == null,此时需要把sHsT同时初始化:

if (sH == null) {
sH = head;
sT = head;
}

如果不是第一次构造,则

sT.next = head;
sT = head;

开始区域的尾指针直接指向当前结点,然后把尾指针设置未当前结点即可。

其他两个区域的链表处理方式同理。

最后需要把三个区域的链表连接起来:

        // 小于区域的尾巴,连等于区域的头,等于区域的尾巴连大于区域的头
if (sT != null) { // 如果有小于区域
sT.next = eH;
eT = eT == null ? sT : eT; // 下一步,谁去连大于区域的头,谁就变成eT
}
// all reconnect
if (eT != null) { // 如果小于区域和等于区域,不是都没有
eT.next = mH;
}
// 如果小于区域有,小于区域的头就是最终链表的头
// 如果小于区域没有,等于区域的头有,则等于区域的头就是最终链表的头
// 如果小于和等于区域都没有,则大于区域的头就是最终链表的头
return sH != null ? sH : (eH != null ? eH : mH);

完整代码见

public class Code_PartitionList {

    public static class ListNode {
public int val;
public ListNode next; public ListNode(int data) {
this.val = data;
}
} public static ListNode listPartition2(ListNode head, int pivot) {
ListNode sH = null; // small head
ListNode sT = null; // small tail
ListNode eH = null; // equal head
ListNode eT = null; // equal tail
ListNode mH = null; // big head
ListNode mT = null; // big tail
ListNode next; // save next node
// every node distributed to three lists
while (head != null) {
next = head.next;
head.next = null;
if (head.val < pivot) {
if (sH == null) {
sH = head;
sT = head;
} else {
sT.next = head;
sT = head;
}
} else if (head.val == pivot) {
if (eH == null) {
eH = head;
eT = head;
} else {
eT.next = head;
eT = head;
}
} else {
if (mH == null) {
mH = head;
mT = head;
} else {
mT.next = head;
mT = head;
}
}
head = next;
}
// 小于区域的尾巴,连等于区域的头,等于区域的尾巴连大于区域的头
if (sT != null) { // 如果有小于区域
sT.next = eH;
eT = eT == null ? sT : eT; // 下一步,谁去连大于区域的头,谁就变成eT
}
// all reconnect
if (eT != null) { // 如果小于区域和等于区域,不是都没有
eT.next = mH;
}
// 如果小于区域有,小于区域的头就是最终链表的头
// 如果小于区域没有,等于区域的头有,则等于区域的头就是最终链表的头
// 如果小于和等于区域都没有,则大于区域的头就是最终链表的头
return sH != null ? sH : (eH != null ? eH : mH);
}
}

解决了链表的荷兰国旗问题,那么原题中的链表 Partition 问题,就迎刃而解了。

由于是 Partition,所以不存在等于区域,只需要考虑小于区域和大于区域,只需要设置如下几个变量即可。

ListNode sH = null; // 小于区域的头
ListNode sT = null; // 小于区域的尾
ListNode mH = null; // 大于区域的头
ListNode mT = null; // 大于区域的尾
ListNode cur = head; // 当前遍历到的结点

接下来开始遍历链表,进行主流程处理,伪代码如下

while (cur != null) {
// 如果head.val < pivot,则通过sH,sT构造小于区域的链表
// 如果head.val > pivot,则通过mH,mT构造小于区域的链表
cur = cur.next;
}
// 把两个区域的链表串联起来即可。

构造每个区域的链表的时候和上述荷兰国旗问题一样。

最后需要把两个区域的链表连接起来:


if (mT != null) {
// 大于区域的尾置空
mT.next = null;
} if (sT != null) {
// 小于区域的尾置空
sT.next = null;
}
// 经过上述操作,两个链表断开连接
if (sH == null) {
// 小于区域的头为空,说明只有大于区域
return mH;
}
// 小于区域的头不为空,小于区域的尾一定也不为空,直接把小于区域的尾连上大于区域的头即可。
sT.next = mH;
return sH;

完整代码见

class Solution {
public static ListNode partition(ListNode head, int x) {
ListNode sH = null;
ListNode sT = null;
ListNode mH = null;
ListNode mT = null;
ListNode cur = head;
while (cur != null) {
if (cur.val < x) {
if (sH == null) {
sH = cur;
} else {
sT.next = cur;
}
sT = cur;
} else {
// cur.val >= x
// 都放到大于等于区域
if (mH == null) {
mH = cur;
} else {
mT.next = cur;
}
mT = cur;
}
cur = cur.next;
}
if (mT != null) {
mT.next = null;
}
if (sT != null) {
sT.next = null;
}
if (sH == null) {
return mH;
}
sT.next = mH;
return sH;
}
}

更多

算法和数据结构笔记

在链表上实现 Partition 以及荷兰国旗问题的更多相关文章

  1. July收集荷兰国旗问题之三路partition

    这道题目和分成两块的partition的扩展.比如有一堆0 1 2 数字组成的数组,要分成 00 00  11 1 1  222 2这样的顺序的. 利用lumoto版的partition能够非常好的解 ...

  2. 快速排序与荷兰国旗及Partition问题

    快速排序与荷兰国旗及Partition问题 需求: 1.Partition过程 给定一个数组arr,和一个整数num.请把小于等于num的数放在数组的左边,大于num的数放在数组的右边. 要求额外空间 ...

  3. 荷兰国旗问题、快排以及BFPRT算法

    荷兰国旗问题 给定一个数组arr,和一个数num,请把小于num的数放数组的左边,等于num的数放在数组的中间,大于num的数放在数组的右边.要求额外空间复杂度O(1),时间复杂度O(N). 这个问题 ...

  4. ACM 荷兰国旗问题

    荷兰国旗问题 时间限制:3000 ms  |  内存限制:65535 KB 难度:1   描述 荷兰国旗有三横条块构成,自上到下的三条块颜色依次为红.白.蓝.现有若干由红.白.蓝三种颜色的条块序列,要 ...

  5. 算法笔记_051:荷兰国旗问题(Java)

    目录 1 问题描述 2 解决方案   1 问题描述 现有n个红白蓝三种不同颜色的小球,乱序排列在一起,请通过两两交换任意两个球,使得从左至右的球依次为红球.白球.蓝球.这个问题之所以叫荷兰国旗,是因为 ...

  6. Coursera Algorithms week2 基础排序 练习测验: Dutch national flag 荷兰国旗问题算法

    第二周课程的Elementray Sorts部分练习测验Interview Questions的第3题荷兰国旗问题很有意思.题目的原文描述如下: Dutch national flag. Given ...

  7. NYOJ_268_荷兰国旗问题

    荷兰国旗问题 时间限制:3000 ms  |  内存限制:65535 KB 难度:1 描写叙述 荷兰国旗有三横条块构成,自上到下的三条块颜色依次为红.白.蓝.现有若干由红.白.蓝三种颜色的条块序列.要 ...

  8. Java实现荷兰国旗问题

    问题描述 现有n个红白蓝三种不同颜色的小球,乱序排列在一起,请通过两两交换任意两个球,使得从左至右的球依次为红球.白球.蓝球.这个问题之所以叫荷兰国旗,是因为将红白蓝三色的小球弄成条状物,并有序排列后 ...

  9. 荷兰国旗 Flag of the Kingdom of the Netherlands

    问题描述:现有n个红白蓝三种不同颜色的小球,乱序排列在一起,请通过两两交换任意两个球,使得从左至右的球依次为红球.白球.蓝球.这个问题之所以叫做荷兰国旗,是因为将红白蓝三色的小球弄成条状物,并有序排列 ...

  10. 148 Sort List 链表上的归并排序和快速排序

    在使用O(n log n) 时间复杂度和常数级空间复杂度下,对链表进行排序. 详见:https://leetcode.com/problems/sort-list/description/ Java实 ...

随机推荐

  1. 手把手教你用Java获取IP归属地

    前几个月微信公众号上线了IP归属地的功能,后续知乎.抖音等平台纷纷添加了该功能.如果是国内的用户精确到省份,国外用户精确到国家.本文就使用Java实现获取IP归属地. ! 主要讲解几个步骤: Java ...

  2. 一步步搞懂MySQL元数据锁(MDL)

    某日,路上收到用户咨询,为了清除空间,想删除某200多G大表数据,且已经确认此表不再有业务访问,于是执行了一条命令'delete from bigtable',但好长时间也没删完,经过咨询后,获知dr ...

  3. Dapr 证书过期了怎么办? 别慌,有救!

    一.背景 Dapr 默认证书有效时间是1年,证书过期后就不能执行相关控制面和数据面的交互了,如下图: 二.查看证书有效时间 通过dapr mtls expiry 看到期时间,具体参见命令https:/ ...

  4. Typora Markdown 安装包

    下载地址: 链接:https://pan.baidu.com/s/1wy0Ik95AjM5WjSC3nzOzqA 提取码:f26j 复制这段内容后打开百度网盘手机App,操作更方便哦 已更新至最新版0 ...

  5. 对于Java中权限修饰符的理解

    老是把Java中权限修饰符给忘记,写一个博客加深印象吧 权限分为四个作用域:当前类,同一个包,其他包的子类,其他包的类. 首先要知道包的概念,Java中一个包是指一个package下的所有文件. pr ...

  6. ProxySQL(8):SQL语句的重写规则

    文章转载自: https://www.cnblogs.com/f-ck-need-u/p/9309760.html 为什么要重写SQL语句 ProxySQL在收到前端发送来的SQL语句后,可以根据已定 ...

  7. SNI 路由和多协议端口的 TCP

    文章转载自:https://mp.weixin.qq.com/s/nMMN7hAJK6SFn1V1YyxvHA 下面是一个简单的示例配置 - 使用最新支持的 YAML 文件格式,将请求路由到一个数据库 ...

  8. Java泛型的总结

    泛型可以用于接口.类.方法上.还有泛型通配符这个概念 泛型的好处:可以在编译时检查 1.用于方法中,指定该方法中的形参的类型. 语法:修饰符 <代表泛型的变量> 返回值类型 方法名(参数) ...

  9. Tubian-Win上线!Tubian官方的Windows软件适配项目

    Sourceforge.net下载:https://sourceforge.net/projects/tubian/ 123网盘下载: https://www.123pan.com/s/XjkKVv- ...

  10. 通过netty把百度地图API获取的地理位置从Android端发送到Java服务器端

    本篇记录我在实现时的思考过程,写给之后可能遇到困难的我自己也给到需要帮助的人. 写的比较浅显,见谅. 在写项目代码的时候,需要把Android端的位置信息传输到服务器端,通过Netty达到连续传输的效 ...