leetcode 75.颜色分类(荷兰国旗问题)

news/2025/3/19 19:28:45/

题目描述

在这里插入图片描述

题目分析

本题是经典的「荷兰国旗问题」,由计算机科学家 Edsger W. Dijkstra 首先提出。
要想单独解决这道题本身还是很简单的,统计0、1、2的数量然后按顺序赋值,或者手写一个冒泡排序,whatever。
但是在这一题中我们主要学习它的思想,题目想要将一个数组分成三个区间,那么就要用到三指针法。时间复杂度O(n)。
在本文的最后会附上leetcode的双指针法,但是本文的主要目的就是学会三指针法,不管哪种方式,核心思想是差不多的,本题的思想同样也是快速排序中对于大量重复元素的优化方式,甚至可以直接作为快排的核心算法

讲解算法原理

既然想要将一个数组分为0、1、2三个区域,那就定义三个指针来解决这个问题。
指针意义:
i 指向数组第一个元素的位置用来遍历数组
left 指向数组0区间最后一个元素
right 指向数组2区间第一个元素

整个数组在完成分区之后的指针位置分布是这样的,数组由三个部分字组成:

在这里插入图片描述
注:x 为0区间最后一个元素, y 为2区间第一个元素

整个数组在正在分区的过程中时的指针位置分布是这样的,数组由四个部分组成:

在这里插入图片描述

我们主要就是要研究正在分区时的指针分布,来规定循环条件以及各种操作。
数组在分区时分为四个区间:
[0, left] [left + 1, i - 1] [i, right -1] [right, n]
从左到右依次是:0区间,1区间,待扫描区域,2区间
当待扫描区域消失后,数组分区就完成了

那么在最刚开始各个指针是怎么分布的呢?

以示例一为例:
nums = [2,0,2,1,1,0]

指针的初始化

在这里插入图片描述

在最刚开始,数组并没有0区间和2区间,所以left指向数组第一个元素的左边位置,right指向数组的最后一个元素的右边位置,i指向数组的第一个元素。

i 指针扫描到不同数据的分类处理1

  1. 当i扫描到0时,将i位置的元素与left + 1位置的元素交换,然后再left++,就将0放到了0区间中,最后再i++,准备扫描下一个元素。
    Q:为什么不直接与left交换?
    A:我们不能直接与left交换,因为left在刚开始是不在数组范围内的。我们是通过将left扩容的方式来涵盖0这个元素。
  2. 当i扫描到2时,将i位置的元素与right - 1位置的元素交换,然后再right–,就将2放到了2区间中,此时i还不能急着++,因为新换过来的元素还没有扫描。就这样进入到下一次循环中,i会再处理这个新换来的元素。
  3. 当i扫描到1时,直接i++,1就进入了1区间。
    请添加图片描述
    请添加图片描述
class Solution {
public:void sortColors(vector<int>& nums) {int left = -1,right = nums.size(); //left为0区间的最后一个元素位置,right为2区间的第一个元素位置int i = 0; //i指针遍历整个数组while(i < right) //[0,left],[left+1,i-1],[i,right-1],[right,n]{if(nums[i] == 0) {swap(nums[i],nums[left+1]);left++;i++;}else if(nums[i] == 1) i++;else {swap(nums[i],nums[right]);right--;}}  }
};

简写后:

class Solution {
public:void sortColors(vector<int>& nums) {int left = -1,right = nums.size(); //left为0区间的最后一个元素位置,right为2区间的第一个元素位置int i = 0; //i指针遍历整个数组while(i < right) //[0,left],[left+1,i-1],[i,right-1],[right,n]{if(nums[i] == 0) swap(nums[i++],nums[++left]);else if(nums[i] == 1) i++;else swap(nums[i],nums[--right]);}  }
};

补充:
leetcode官方给出的双指针法也是经典的方法,在此引用供大家学习:
在这里插入图片描述
在这里插入图片描述
题解作者:力扣官方
题解链接:https://leetcode.cn/problems/sort-colors/solutions/437968/yan-se-fen-lei-by-leetcode-solution/


http://www.ppmy.cn/news/1580399.html

相关文章

台式机电脑组装---电源

台式机电脑组装—电源 22 33 主板供电是聚集了12V&#xff0c;5V,3.3V的24pin CPU供电的话主要是12V的44pin供电 44pin合并之后&#xff0c;就是8pin 55 SATA硬盘会使用饼io口取电&#xff0c;从电源获取12v,5v,3.3v的电 33

3--网络安全架构概述

从青铜到王者&#xff1a;华为网络安全架构全景解读 前言&#xff1a;黑客都开始内卷了&#xff0c;你的网络安全还在裸奔吗&#xff1f; “从前黑客攻击是为了炫技&#xff0c;现在攻击是为了还房贷” —— 某不愿透露姓名的白帽子 当网络攻击从"技术宅的恶作剧"变…

面试redis常被问到的面试题含答案

什么是Redis&#xff1f;它的特点是什么&#xff1f; Redis是一个开源的内存数据库&#xff0c;用于存储数据并支持多种数据结构&#xff08;如字符串、哈希、列表、集合、有序集合等&#xff09;。其特点包括高性能、支持持久化、数据结构丰富、原子性操作、支持事务等。 Red…

vue网格布局--grid布局

1 九宫格布局&#xff08;无边距&#xff09; <div class"container"><div class"item">1</div><div class"item">2</div><div class"item">3</div><div class"item">4<…

外星人入侵-Python-三

武装飞船 开发一个名为《外星人入侵》的游戏吧&#xff01;为此将使用 Pygame&#xff0c;这是一组功能强大而有趣的模块&#xff0c;可用于管理图形、动画乃至声音&#xff0c; 让你能够更轻松地开发复杂的游戏。通过使用Pygame来处理在屏幕上绘制图像 等任务&#xff0c;可将…

音视频处理的“瑞士军刀”与“积木”:FFmpeg 与 GStreamer 的深度揭秘

一、发展历史与生态演进对比 FFmpeg的成长轨迹 诞生背景&#xff1a;2000年由Fabrice Bellard创建&#xff0c;最初为解决视频编码标准化问题而生。早期版本仅支持MPEG-1编码&#xff0c;但凭借开源社区协作&#xff0c;迅速扩展为全格式编解码工具。技术扩张&#xff1a;2004年…

matlab 火电厂给水控制系统仿真

1、内容简介 略 matlab157-火电厂给水控制系统仿真 可以交流、咨询、答疑 2、内容说明 略 摘 要 虽然现在新能源发电领域比较火爆&#xff0c;但至今火力发电厂依然在我的的发电领域中拥有很重要的地位。我国虽然还是发展中国家&#xff0c;但是近年来GDP的增长已经处于世界…

【AI】利用Azure AI的元数据过滤器提升 RAG 性能并增强向量搜索案例

【AI】利用Azure AI的元数据过滤器提升 RAG 性能并增强向量搜索案例 在检索增强生成 (RAG) 设置中,用户指定的过滤器(无论是隐含的还是明确的)通常在向量搜索中被忽视,因为向量搜索主要关注语义相似性。 在某些场景中,确保特定的查询仅使用预定义的(子)文档集来回答至关…