分隔链表

题目描述

CategoryDifficultyLikesDislikes
algorithmsMedium (64.44%)851-

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

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

示例 1:

https://assets.leetcode.com/uploads/2021/01/04/partition.jpg

输入:head = [1,4,3,2,5,2], x = 3
输出:[1,2,2,4,3,5]

示例 2:

输入:head = [2,1], x = 2
输出:[1,2]

提示:

  • 链表中节点的数目在范围 [0, 200]
  • 100 <= Node.val <= 100
  • 200 <= x <= 200

求解

方法一:模拟

思路及算法

直观来说我们只需维护两个链表 small 和 large 即可,small 链表按顺序存储所有小于 x 的节点,large 链表按顺序存储所有大于等于 x 的节点。遍历完原链表后,我们只要将 small 链表尾节点指向 large 链表的头节点即能完成对链表的分隔。

为了实现上述思路,我们设 smallHead 和 largeHead 分别为两个链表的哑节点,即它们的 next 指针指向链表的头节点,这样做的目的是为了更方便地处理头节点为空的边界条件。同时设 small 和 large 节点指向当前链表的末尾节点。开始时 smallHead=small,largeHead=large。随后,从前往后遍历链表,判断当前链表的节点值是否小于 x,如果小于就将 small 的 next 指针指向该节点,否则将 large 的 next 指针指向该节点。

遍历结束后,我们将 large 的 next 指针置空,这是因为当前节点复用的是原链表的节点,而其 next 指针可能指向一个小于 x 的节点,我们需要切断这个引用。同时将 small 的 next 指针指向 largeHead 的 next 指针指向的节点,即真正意义上的 large 链表的头节点。最后返回 smallHead 的 next 指针即为我们要求的答案。

代码

class Solution
{
public:
    ListNode *partition(ListNode *head, int x)
    {
        ListNode *smallHead = new ListNode();
        ListNode *small = smallHead;
        ListNode *largeHead = new ListNode();
        ListNode *large = largeHead;

        while (head)
        {
            if (head->val < x)
            {
                small->next = head;
                small = small->next;
            }
            else
            {
                large->next = head;
                large = large->next;
            }
            head = head->next;
        }
        small->next = largeHead->next;
        large->next = nullptr;

        ListNode *ans = smallHead->next;
        delete smallHead;
        delete largeHead;
        return ans;
    }
};

复杂度分析

  • 时间复杂度: O(n),其中 n 是原链表的长度。我们对该链表进行了一次遍历。
  • 空间复杂度: O(1)。

方法二:队列

队列具有先进先出的特点,我们可以创建两个队列,分别为queue_less(存储小于x的节点指针), queue_more(存储大于x的节点指针),这样既可以分隔节点而且保留了原始链表的节点顺序。

先遍历一遍链表将节点分别存储到这两个队列中。然后创建建一个哑节点,先取出queue_less中的节点一个一个连接,再取出queue_more中的节点一个一个连接。最后将最终节点的next置空防止产生环形节点。

代码

class Solution
{
public:
    ListNode *partition(ListNode *head, int x)
    {
        queue<ListNode *> queue_less, queue_more;
        ListNode *dummy = new ListNode(0, head);
        ListNode *p = dummy->next;

        while (p)
        {
            if (p->val < x)
                queue_less.push(p);
            else
                queue_more.push(p);

            p = p->next;
        }

        p = dummy;

        while (!queue_less.empty())
        {
            p->next = queue_less.front();
            queue_less.pop();
            p = p->next;
        }
        while (!queue_more.empty())
        {
            p->next = queue_more.front();
            queue_more.pop();
            p = p->next;
        }
        p->next = nullptr;

        ListNode *ans = dummy->next;
        delete dummy;
        return ans;
    }
};

复杂度分析

  • 时间复杂度: O(n),其中 n 是原链表的长度。我们对该链表进行了一次遍历。
  • 空间复杂度: O(n),借助队列存储 n 个节点的指针。
  • 12
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
好的,我理解了您的问题。下面是C++代码实现: ```c++ #include <iostream> #include <vector> using namespace std; struct ListNode { int val; ListNode *next; ListNode(int x) : val(x), next(NULL) {} }; vector<ListNode*> splitListToParts(ListNode* head, int k) { vector<ListNode*> res(k, nullptr); int len = 0; for (ListNode* p = head; p; p = p->next) ++len; int n = len / k, r = len % k; ListNode* p = head, *prev = nullptr; for (int i = 0; p && i < k; ++i, --r) { res[i] = p; for (int j = 0; j < n + (r > 0); ++j) { prev = p; p = p->next; } prev->next = nullptr; } return res; } int main() { // 创建链表 ListNode* head = new ListNode(1); head->next = new ListNode(2); head->next->next = new ListNode(3); head->next->next->next = new ListNode(4); head->next->next->next->next = new ListNode(5); // 分隔链表 vector<ListNode*> parts = splitListToParts(head, 3); for (int i = 0; i < parts.size(); ++i) { cout << "Part " << i + 1 << ": "; ListNode* p = parts[i]; while (p) { cout << p->val << " "; p = p->next; } cout << endl; } return 0; } ``` 这段代码中,我们使用了 `vector` 来存储分隔后的链表。首先计算链表长度 `len`,然后计算每个部分的长度 `n = len / k` 和余数 `r = len % k`。接下来我们遍历链表,将每个部分的头结点存储到 `vector` 中,然后将链表断开,重复上述步骤,最后返回 `vector` 即可。 运行结果如下: ``` Part 1: 1 2 Part 2: 3 4 Part 3: 5 ``` 这里我们将长度尽可能相等的策略应用到了每个部分中,因此部分 1 和 2 的长度相等,部分 3 的长度比它们短 1。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

泡椒香菜‍

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值