【程序员面试算法】快速排序,9种主流编程语言实现!

快速排序是一种高效的排序算法,基于分治策略。是面试中频繁考察的重点之一。
原理:它选择一个基准元素,通过一趟排序将待排序序列分割成两部分。其中一部分的所有元素都比基准元素小,另一部分则比基准元素大。然后,对这两部分分别进行快速排序,整个序列就逐步变得有序。
实现思路:

  1. 首先,在序列中选择一个基准元素,常见的是选择第一个或最后一个元素。
  2. 接着,从序列的两端开始设置两个指针,一个从左向右,一个从右向左。左指针寻找大于基准的元素,右指针寻找小于基准的元素,当找到后交换这两个元素。
  3. 持续这个过程,直到两个指针相遇,此时将基准元素与相遇位置的元素交换,这样就完成了一次划分。
  4. 递归地对划分后的两个子序列重复上述操作,直到子序列的长度为 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)
}

以上代码分别展示了不同编程语言实现非递归的快速排序算法。希望这篇文章能对你有所帮助,记得收藏起来哦,方便随时查阅和复习。

posted @ 2024-11-22 07:50  Newton爱编程  阅读(32)  评论(0编辑  收藏  举报