【程序员面试算法】快速排序,9种主流编程语言实现!
快速排序是一种高效的排序算法,基于分治策略。是面试中频繁考察的重点之一。
原理:它选择一个基准元素,通过一趟排序将待排序序列分割成两部分。其中一部分的所有元素都比基准元素小,另一部分则比基准元素大。然后,对这两部分分别进行快速排序,整个序列就逐步变得有序。
实现思路:
- 首先,在序列中选择一个基准元素,常见的是选择第一个或最后一个元素。
- 接着,从序列的两端开始设置两个指针,一个从左向右,一个从右向左。左指针寻找大于基准的元素,右指针寻找小于基准的元素,当找到后交换这两个元素。
- 持续这个过程,直到两个指针相遇,此时将基准元素与相遇位置的元素交换,这样就完成了一次划分。
- 递归地对划分后的两个子序列重复上述操作,直到子序列的长度为 1 或 0,此时整个序列排序完成。它的平均时间复杂度为O(n log n) ,但在最坏情况下可能退化为 O(n2)。
本文使用 C、C++ 、Python、Java、Kotlin、Swift、JavaScript、Rust 和 Go 9 种主流编程语言实现快速排序算法。
1. C语言
#include <stdio.h>
// 交换两个元素的值
void swap(int* a, int* b) {
int temp = *a;
*a = *b;
*b = temp;
}
// 核心代码,理解了这个函数就理解了整个算法。
// 划分函数1,以最后一个元素为基准,将小于基准的元素放在左边,大于基准的元素放在右边
int partition(int arr[], int low, int high) {
int pivot = arr[low];
int start = low;
while (low < high) {
while (low < high && arr[high] >= pivot) {
high--;
}
while (low < high && arr[low] <= pivot) {
low++;
}
swap(&arr[low], &arr[high]);
}
swap(&arr[start], &arr[low]);
return low;
}
// 划分函数2,其它语言参考该方法即可
int partition(int arr[], int low, int high) {
int pivot = arr[high]; // 选择最后一个元素作为基准
int i = low - 1; // i 是小于基准的元素的索引
for (int j = low; j < high; j++) {
if (arr[j] < pivot) {
i++;
swap(&arr[i], &arr[j]);
}
}
swap(&arr[i + 1], &arr[high]); // 关键,将基准元素放到正确位置
return i + 1;
}
// 快速排序函数
void quickSort(int arr[], int low, int high) {
if (low < high) {
int pi = partition(arr, low, high);
quickSort(arr, low, pi - 1); // 递归排序左半部分
quickSort(arr, pi + 1, high); // 递归排序右半部分
}
}
// 打印数组
void printArray(int arr[], int size) {
for (int i = 0; i < size; i++) {
printf("%d ", arr[i]);
}
}
int main() {
int arr[] = {10, 7, 8, 9, 1, 5};
int n = sizeof(arr) / sizeof(arr[0]);
printf("原始数组:\n");
printArray(arr, n);
quickSort(arr, 0, n - 1);
printf("排序后的数组:\n");
printArray(arr, n);
return 0;
}
2. C++
#include <iostream>
using namespace std;
// 划分函数1
int partition(vector<int>& nums, int low, int high) {
int pivot = nums[low];
int start = low;
while (low < high) {
while (low < high && nums[high] >= pivot) high--;
while (low < high && nums[low] <= pivot) low++;
if (low >= high) {
break;
}
swap(nums[low], nums[high]);
}
swap(nums[start], nums[low]);
return low;
}
// 划分函数2
int partition(vector<int>& arr, int low, int high) {
int pivot = arr[high];
int i = low - 1;
for (int j = low; j < high; j++) {
if (arr[j] < pivot) {
i++;
swap(arr[i], arr[j]);
}
}
swap(arr[i + 1], arr[high]);
return i + 1;
}
// 快速排序函数
void quickSort(vector<int>& arr, int low, int high) {
if (low < high) {
int pi = partition(arr, low, high);
quickSort(arr, low, pi - 1);
quickSort(arr, pi + 1, high);
}
}
int main() {
vector<int> arr = {10, 7, 8, 9, 1, 5};
cout << "原始数组:";
for (int num : arr) cout << num << " ";
cout << endl;
quickSort(arr, 0, arr.size() - 1);
cout << "排序后的数组:";
for (int num : arr) cout << num << " ";
cout << endl;
return 0;
}
3. Python
def swap(arr, i, j):
arr[i], arr[j] = arr[j], arr[i]
# 划分函数1
def partition(arr, low, high):
pivot = arr[low]
start = low
while low < high:
while low < high and arr[high] >= pivot:
high -= 1
while low < high and arr[low] <= pivot:
low += 1
if low >= high:
break
swap(arr, low, high)
swap(arr, start, low)
return low
# 划分函数2
def partition(arr, low, high):
pivot = arr[high]
i = low - 1
for j in range(low, high):
if arr[j] < pivot:
i += 1
swap(arr, i, j)
swap(arr, i + 1, high)
return i + 1
def quick_sort(arr):
low, high = 0, len(arr) - 1
while low < high:
pivot = partition(arr, low, high)
quick_sort(arr[low:pivot])
quick_sort(arr[pivot + 1:high + 1])
arr = [10, 7, 8, 9, 1, 5]
print("原始数组:", arr)
quick_sort(arr)
print("排序后的数组:", arr)
4. Java
递归
public class QuickSortJava {
static void swap(int[] arr, int i, int j) {
int temp = arr[i];
arr[i] = arr[j];
arr[j] = temp;
}
// 划分函数1
static int partition(int[] nums, int low, int high) {
int pivot = nums[low]; // 选定数组第一个元素作为基准值
int start = low; // 记录基准值的初始位置
while (low < high) {
// 从右向左找第一个小于基准值的元素
while (low < high && nums[high] >= pivot) {
high--;
}
// 从左向右找第一个大于基准值的元素
while (low < high && nums[low] <= pivot) {
low++;
}
// 如果low和high没有交错,交换它们找到的元素
if (low >= high) {
break;
}
swap(nums, low, high);
}
// 基准值归位:将基准值放到low(或high,此时它们相等)的位置上
swap(nums, start, low);
// 返回基准值的最终位置
return low;
}
// 划分函数2
static int partition(int[] arr, int low, int high) {
int pivot = arr[high];
int i = low - 1;
for (int j = low; j < high; j++) {
if (arr[j] < pivot) {
i++;
swap(arr, i, j);
}
}
int temp = arr[i + 1];
arr[i + 1] = arr[high];
arr[high] = temp;
return i + 1;
}
static void quickSort(int[] arr, int low, int high) {
if (low < high) {
int pi = partition(arr, low, high);
quickSort(arr, low, pi - 1);
quickSort(arr, pi + 1, high);
}
}
static void main(String[] args) {
int[] arr = {10, 7, 8, 9, 1, 5};
quickSort(arr, 0, arr.length - 1);
System.out.println(Arrays.toString(arr));
}
}
非递归
import java.util.Stack;
class Solution {
public int[] quickSort(int[] nums) {
Stack<Integer> stack = new Stack<>();
stack.push(nums.length - 1);
stack.push(0);
while (!stack.isEmpty()) {
int low = stack.pop();
int high = stack.pop();
if (low < high) {
int index = partition(nums, low, high);
// Push right, middle, and left indices in correct order
stack.push(index - 1);
stack.push(low);
stack.push(high);
stack.push(index + 1);
}
}
return nums;
}
private int partition(int[] nums, int low, int high) {
int pivot = nums[low];
int start = low;
while (low < high) {
while (low < high && nums[high] >= pivot) {
high--;
}
while (low < high && nums[low] <= pivot) {
low++;
}
if (low >= high) {
break;
}
swap(nums, low, high);
}
swap(nums, start, low); // 关键,将基准元素放到正确位置
return low;
}
private void swap(int[] nums, int i, int j) {
int temp = nums[i];
nums[i] = nums[j];
nums[j] = temp;
}
}
5. Kotlin
fun swap(arr: IntArray, i: Int, j: Int) {
val temp = arr[i]
arr[i] = arr[j]
arr[j] = temp
}
// 划分函数1
fun partition(nums: IntArray, low: Int, high: Int): Int {
val pivot = nums[low]
var start = low
while (low < high) {
while (low < high && nums[high] >= pivot) {
high--
}
while (low < high && nums[low] <= pivot) {
low++
}
if (low >= high) {
break
}
swap(nums, low, high)
}
swap(nums, start, low)
return low
}
// 划分函数2
fun partition(arr: IntArray, low: Int, high: Int): Int {
val pivot = arr[high]
var i = low - 1
for (j in low until high) {
if (arr[j] < pivot) {
i++
swap(arr, i, j)
}
}
swap(arr, i + 1, high)
return i + 1
}
fun quickSort(arr: IntArray) {
var low = 0
var high = arr.size - 1
while (low < high) {
val pivot = partition(arr, low, high)
quickSort(arr.copyOfRange(low, pivot))
quickSort(arr.copyOfRange(pivot + 1, high + 1))
}
}
fun main() {
val arr = intArrayOf(10, 7, 8, 9, 1, 5)
println("原始数组: ${arr.joinToString()}")
quickSort(arr)
println("排序后的数组: ${arr.joinToString()}")
}
6. Swift
func partition(_ nums: inout [Int], low: Int, high: Int) -> Int {
let pivot = nums[low]
var start = low
while low < high {
while low < high && nums[high] >= pivot {
high -= 1
}
while low < high && nums[low] <= pivot {
low += 1
}
if low >= high {
break
}
swap(&nums, low, high)
}
swap(&nums, start, low)
return low
}
func partition(_ arr: inout [Int], low: Int, high: Int) -> Int {
let pivot = arr[high]
var i = low - 1
for j in low..<high {
if arr[j] < pivot {
i += 1
swap(&arr, i, j)
}
}
swap(&arr, i + 1, high)
return i + 1
}
func quickSort(_ arr: inout [Int]) {
var low = 0
var high = arr.count - 1
while low < high {
let pivot = partition(&arr, low: low, high: high)
if pivot - low < high - pivot {
quickSort(&arr[low..<pivot])
low = pivot + 1
} else {
quickSort(&arr[(pivot + 1)...high])
high = pivot - 1
}
}
}
var arr = [10, 7, 8, 9, 1, 5]
print("原始数组: \(arr)")
quickSort(&arr)
print("排序后的数组: \(arr)")
7. JavaScript
function swap(arr, i, j) {
[arr[i], arr[j]] = [arr[j], arr[i]];
}
function partition(nums, low, high) {
let pivot = nums[low];
let start = low;
while (low < high) {
while (low < high && nums[high] >= pivot) {
high--;
}
while (low < high && nums[low] <= pivot) {
low++;
}
if (low >= high) {
break;
}
swap(nums, low, high);
}
swap(nums, start, low);
return low;
}
function partition(arr, low, high) {
const pivot = arr[high];
let i = low - 1;
for (let j = low; j < high; j++) {
if (arr[j] < pivot) {
i++;
swap(arr, i, j);
}
}
swap(arr, i + 1, high);
return i + 1;
}
function quickSort(arr) {
let low = 0;
let high = arr.length - 1;
while (low < high) {
const pivot = partition(arr, low, high);
if (pivot - low < high - pivot) {
quickSort(arr.slice(low, pivot));
low = pivot + 1;
} else {
quickSort(arr.slice(pivot + 1, high + 1));
high = pivot - 1;
}
}
}
const arr = [10, 7, 8, 9, 1, 5];
console.log("原始数组:", arr);
quickSort(arr);
console.log("排序后的数组:", arr);
8. Rust
fn swap(arr: &mut Vec<i32>, i: usize, j: usize) {
arr.swap(i, j);
}
fn partition(nums: &mut [i32], low: usize, high: usize) -> usize {
let pivot = nums[low];
let mut start = low;
let mut low = low;
let mut high = high;
while low < high {
while low < high && nums[high] >= pivot {
high -= 1;
}
while low < high && nums[low] <= pivot {
low += 1;
}
if low >= high {
break;
}
nums.swap(low, high);
}
nums.swap(start, low);
low
}
fn partition(arr: &mut Vec<i32>, low: usize, high: usize) -> usize {
let pivot = arr[high];
let mut i = low as isize - 1;
for j in low..high {
if arr[j] < pivot {
i += 1;
swap(arr, i as usize, j);
}
}
swap(arr, (i + 1) as usize, high);
(i + 1) as usize
}
fn quick_sort(arr: &mut Vec<i32>) {
let mut low = 0;
let mut high = arr.len() - 1;
while low < high {
let pivot = partition(arr, low, high);
quick_sort(&mut arr[low..pivot].to_vec());
quick_sort(&mut arr[(pivot + 1)..=high].to_vec());
}
}
fn main() {
let mut arr = vec![10, 7, 8, 9, 1, 5];
println!("原始数组: {:?}", arr);
quick_sort(&mut arr);
println!("排序后的数组: {:?}", arr);
}
9. Go
package main
import (
"fmt"
)
func swap(arr []int, i, j int) {
arr[i], arr[j] = arr[j], arr[i]
}
func partition(nums []int, low, high int) int {
pivot := nums[low]
start := low
for low < high {
for low < high && nums[high] >= pivot {
high--
}
for low < high && nums[low] <= pivot {
low++
}
if low >= high {
break
}
swap(nums, low, high)
}
swap(nums, start, low)
return low
}
func partition(arr []int, low, high int) int {
pivot := arr[high]
i := low - 1
for j := low; j < high; j++ {
if arr[j] < pivot {
i++
swap(arr, i, j)
}
}
swap(arr, i+1, high)
return i + 1
}
func quickSort(arr []int) {
low := 0
high := len(arr) - 1
for low < high {
pivot := partition(arr, low, high)
quickSort(arr[low:pivot])
quickSort(arr[pivot+1 : high+1])
}
}
func main() {
arr := []int{10, 7, 8, 9, 1, 5}
fmt.Println("原始数组:", arr)
quickSort(arr)
fmt.Println("排序后的数组:", arr)
}
以上代码分别展示了不同编程语言实现非递归的快速排序算法。希望这篇文章能对你有所帮助,记得收藏起来哦,方便随时查阅和复习。