【Golang】LeetCode 75. 颜色分类

75. 颜色分类

题目描述

思路

看到"相同的颜色相邻"、"原地修改数组"、"对数组原地排序"的字样,我第一眼想到的思路就是使用双指针。在 Hot 100 的双指针专题当中,我们已经接触过「移动零」这样的题目了,这道题其实就是「移动零」当中双指针的应用。

我们使用两种思路来解决这道题目,分别是「两次遍历」和「一次遍历」。

首先是「两次遍历」。两次遍历我们要做什么?第一次遍历我们要做的就是把数组当中的0放到前面;第二次遍历我们要做的就是将数组当中的1放到0的后面。有了这样的思路,我们可以使用快慢指针来实现需求。具体来说,令i是慢指针,j是快指针,如果nums[j] == 0,我们就交换nums[i]nums[j],并令i += 1。第一次遍历之后,[0, i)的左闭右开区间内都是0。我们接着从i的位置开始找[i + 1, n)子数组当中的1。思路和找0相同,如果遇到nums[j] == 1,就和nums[i]交换,并令i += 1。这样,通过两次遍历,我们可以得到题目中要求的结果。

其次是「一次遍历」。「两次遍历」的思路非常直观,但我们自然能够想到,如果能够建立三个指针p0 / p1 / j,是不是通过一次遍历就可以完成数组的排序?答案是可以的,这也是「题目描述」当中"进阶"部分要求我们实现的思路。具体来说,如果nums[j] == 1,我们直接交换nums[p1]nums[j]即可,并令p1 += 1。而如果nums[j] == 0,我们不能仅仅简单地交换nums[p0]nums[j],原因是p0所指的位置可能已经填过1了,因此我们需要判断一道p0p1的大小关系。如果p0 < p1,说明这个位置之前已经放过1了,由于nums[p0]nums[j]交换使j的位置放置了1,我们只需要再对nums[p1]nums[j]进行一次交换即可。最后同时移动p0p1。这样通过一次遍历,我们就能解决问题。

Golang 题解

「两次遍历」

go 复制代码
func sortColors(nums []int)  {
    n := len(nums)
    // 双指针, 两次遍历
    i := 0
    for j := 0; j < n; j ++ {
        if nums[j] == 0 {
            nums[i], nums[j] = nums[j], nums[i]
            i ++
        }
    }
    for j := i; j < n; j ++ {
        if nums[j] == 1 {
            nums[i], nums[j] = nums[j], nums[i]
            i ++
        }
    }
}

「一次遍历」

go 复制代码
func sortColors(nums []int)  {
    n := len(nums)
    p0, p1 := 0, 0
    for j := 0; j < n; j ++ {
        if nums[j] == 1 {
            nums[j], nums[p1] = nums[p1], nums[j]
            p1 ++
        } else if nums[j] == 0 {
            nums[j], nums[p0] = nums[p0], nums[j]
            if p0 < p1 {
                // 如果 p0 < p1, 说明此时 nums[j] 交换得到的数是 1
                nums[j], nums[p1] = nums[p1], nums[j]
            }
            p0 ++
            p1 ++
        }
    }
}
相关推荐
LuminousCPP6 分钟前
数据结构 - 线性表第三篇:基于顺序表实现 C 语言通讯录(基础功能篇)
c语言·数据结构·经验分享·笔记·算法
_日拱一卒10 分钟前
LeetCode:114二叉树展开为链表
java·开发语言·算法
无小道20 分钟前
Redis——哈希类型相关指令
redis·算法·哈希算法
凌波粒22 分钟前
LeetCode--513.找树左下角的值(二叉树)
java·算法·leetcode
一个不知名程序员www22 分钟前
算法学习入门---算法题DAY1
c++·算法
子琦啊27 分钟前
构造函数、this指向和原型链机制
javascript·算法·贴图
WHS-_-202229 分钟前
Millimeter Wave ISAC-SLAM: Framework and RFSoC Prototype
人工智能·算法·原型模式
吃好睡好便好30 分钟前
在Matlab中绘制杆状图
开发语言·学习·算法·matlab·信息可视化
带带弟弟学爬虫__31 分钟前
dyAPP数据采集-个人主页、发布、搜索、评论
服务器·python·算法·flutter·java-ee·django
sali-tec36 分钟前
C# 基于OpenCv的视觉工作流-章75-线-线角度
图像处理·人工智能·opencv·算法·计算机视觉