王清欢Randy 王清欢Randy
首页
  • 编程语言

    • C/C++ 学习笔记
    • Golang 学习笔记
  • 算法分析

    • LeetCode 刷题笔记
  • 操作系统

    • Linux 基础
    • Vim 实用技巧
    • Shell 脚本编程
    • GDB 学习笔记
  • 开发工具

    • Git 学习笔记
  • 分布式理论

    • 共识算法
    • 分布式事务
  • 数据库内核

    • PostgreSQL
    • Postgres-XL
  • hidb
  • pgproxy
  • 实用技巧
  • 学习方法
  • 资源分享
GitHub (opens new window)
首页
  • 编程语言

    • C/C++ 学习笔记
    • Golang 学习笔记
  • 算法分析

    • LeetCode 刷题笔记
  • 操作系统

    • Linux 基础
    • Vim 实用技巧
    • Shell 脚本编程
    • GDB 学习笔记
  • 开发工具

    • Git 学习笔记
  • 分布式理论

    • 共识算法
    • 分布式事务
  • 数据库内核

    • PostgreSQL
    • Postgres-XL
  • hidb
  • pgproxy
  • 实用技巧
  • 学习方法
  • 资源分享
GitHub (opens new window)
  • 基础算法

    • 双指针

      • 双指针基础
      • 碰撞指针
      • 快慢指针
      • 滑动窗口
    • 二分查找

      • 基础应用
      • 边界收缩
      • 局部有序
    • 排序算法

      • 01八大排序
      • 02快速排序
        • 02 快速排序
          • 快速排序简介
          • 215 数组中的第K个最大元素
      • 03归并排序
      • 04桶排序
      • 05堆排序
    • 优先搜索

      • 01递归
      • 02网格结构深度优先搜索
      • 03树结构深度优先搜索
      • 04图结构深度优先搜索
      • 05网格结构广度优先搜索
      • 06树结构广度优先搜索
      • 07图结构广度优先搜索
    • 回溯算法

      • 01递归
      • 02 子集问题
      • 03 全排列问题
      • 04 组合问题
      • 05 回溯搜索问题
  • 基础数据结构

    • 线性表与哈希表

      • 01数组
      • 02栈和队列
      • 03单调栈
      • 04优先队列
      • 05双端队列
      • 06哈希表
      • 07多重集合
      • 08前缀和
      • 09数据结构设计
    • 字符串

      • 01字符串比较
      • 02回文字符串
      • 03字符串匹配
      • 04字符串算术表达式
    • 单链表

      • 01链表基础操作
      • 02链表遍历
    • 二叉树

      • 01二叉树的属性
      • 02二叉树的操作
      • 03层次遍历
      • 04前中后序遍历
      • 05二叉搜索树的属性
      • 06二叉搜索树的操作
      • 07字典树
      • 08二叉搜索树BST
    • 图

      • 01二分图
      • 02拓扑排序
      • 03并查集
      • 04最小生成树
      • 05最短路径
  • 进阶算法

    • 贪心算法

      • 01跳跃游戏
      • 02分配问题
      • 03区间问题
    • 分治策略

    • 动态规划

      • 01一维动态规划
      • 02二维动态规划
      • 03分割型动态规划
      • 04子序列问题
      • 05背包问题
      • 06字符串编辑问题
      • 07股票交易问题
  • 其他内容

    • 数学问题

      • 01公倍数与公因数
      • 02质数问题
      • 03进制转换问题
      • 04数字字符串求和问题
      • 05众数问题
      • 06中位数问题
      • 07数字处理问题
      • 08随机数问题
    • 位运算

      • 01位运算基础
      • 02妙用异或运算
      • 03二进制特性
  • LeetCode刷题笔记
  • 基础算法
  • 排序算法
王清欢
2023-03-24
目录

02快速排序

# 02 快速排序

# 快速排序简介

算法思想:

快速排序是对冒泡排序的一种改进,其核心算法思想是:使用基准将要排序的数据分割成小于基准和大于基准的两部分;然后在被分割的两个部分中递归按基准划分的步骤,最终递归到一个部分仅有一个元素组成,此时数组排序完成。

算法实现过程中,在使用快排之前可以将数组打乱,因为快排是不稳定的算法,在原数组大部分元素是有序的情况下效率提升不明显。

实现快排步骤如下:
  • 先选取基准,可以以序列第一个元素作为基准
  • 然后使用碰撞指针遍历数组,从指向数组尾部的指针 tail 开始移动,将数组尾部小于基准的元素覆盖到指向数组头部的指针 head;接着移动 head 找到大于基准的元素覆盖到 tail;重复该过程直到一次遍历完成,将基准值放到指针相遇位置,将数组划分为小于基准和大于基准的两部分。
  • 在被分割的两部分中递归选择基准和划分序列的步骤,直到递归结束完成排序

执行样例:

输入:[6,1,2,7,9,3,4,5,10,8]

快速排序

算法实现:

// 碰撞指针实现
void quickSort(vector<int> &nums, int l, int r) {
    if (l + 1 >= r) {
    	return;
    }
    int head = l, tail = r - 1, key = nums[head];
    while (head < tail){
        while(head < tail && nums[tail] >= key) {
        	--tail;
        }
        nums[head] = nums[tail];
        while (head < tail && nums[head] <= key) {
        	++head;
        }
        nums[tail] = nums[head];
    }
    nums[head] = key;
    quick_sort(nums, l, head);
    quick_sort(nums, head + 1, r);
}

// 快慢指针实现
void quickSort(vector<int> &nums, int l, int r) {
    if (l + 1 >= r) {
        return;
    }
      
    int cur = l;
    int pre = cur - 1;
    int key = nums[r-1];
      
    while (cur < r) {
        while (nums[cur] < key && ++pre != cur) {
            swap(nums[cur], nums[pre]);
        }
        cur++;
    }   
    swap(nums[++pre], nums[r-1]);
      
    quickSort(nums, l, pre);
    quickSort(nums, pre + 1, r);
}

# 215 数组中的第K个最大元素 (opens new window)

给定整数数组 nums 和整数 k,请返回数组中第 k 个最大的元素。

输入一个数组和一个目标值 k,输出一个整数表示第 k 大的数字。题目默认一定有解。

输入: [3,2,3,1,2,4,5,5,6] 和 k = 4
输出: 4

解析:

一种简单的思路就是将整个数组排完序,然后输出第 k 个最大元素。

我们可以使用快速排序的思想,减少程序的时间复杂度。根据快速排序核心思想衍生出的**快速选择一般用于求解 k-th Element 问题**,可以在 O(n) 时间复杂度,O(1) 空间复杂度完成求解工作。

快速选择的实现和快速排序相似,不过只需要找到第 k 大的枢(pivot)即可,不需要对其左右再进行排序,即在快速排序的基础上增加递归跳出条件,在找到第 k 大的枢时直接终止递归即可,减少时间复杂度。与快速排序一样,快速选择一般需要先打乱数组,否则最坏情况下时间复杂度为 O(n^2)。
class Solution {
public:
    int partation(vector<int>& nums, int k, int start, int end){
        if(start>=end) return -1;
        // 随机化处理
        int idx = start + rand() % (end - start); 
        swap(nums[start], nums[idx]);
        // 一遍遍历与交换
        int head = start, tail = end-1;
        int key = nums[head];
        while(head < tail){
            while(head<tail && nums[tail]>=key){
                --tail;
            }
            nums[head] = nums[tail];
            while(head<tail && nums[head]<=key){
                ++head;
            }
            nums[tail] = nums[head];
        }
        nums[head] = key;
        // 计算当前枢是否为第 k 大的枢
        int a = end - tail;
        // 是第 k 大的枢直接返回
        if(a==k){
            return nums[head];
        }else if(a > k){
            // 在当前枢右侧,递归快速选择
            partation(nums,k,tail+1,end);
        }else{
            // 在当前枢左侧,递归快速选择
            partation(nums,k-a,start,tail);
        }
        return nums[nums.size()-k];
    }

    int findKthLargest(vector<int>& nums, int k) {
        return partation(nums,k,0,nums.size());
    }
};
上次更新: 2023/11/19, 12:55:48
01八大排序
03归并排序

← 01八大排序 03归并排序→

Theme by Vdoing | Copyright © 2023-2024 Wang Qinghuan | MIT License
  • 跟随系统
  • 浅色模式
  • 深色模式
  • 阅读模式