快排归并排序模板

快速排序

快速排序的算法思想是分治,主要有以下四个步骤

  1. 选取一个分界点,比如左端点,右端点或中间值
  2. 调整区间,让分界点左边的值都 < 分界点,分界点右边的值 > 分界点
  3. 递归处理左右区间

其中,比较复杂的一步就是第二步。如何调整区间呢?我们可以使用双指针l、r,分别从左右端点进行移动,当arrl <= 分界点时,继续移动,直到arrl > 分界点;同理,当arrr >= 分界点时,继续移动,直到arrr < 分界点。然后我们再交换arrl,arrr的值,这样就可以调整区间了。

代码

c++

scss 复制代码
void quickSort(int a[], int l, int r){
    //如果数组中就一个数,就已经排好了,直接返回。
    if(l >= r) return;
    //选取分界线。这里选数组中间那个数
    int x = a[l + r >> 1];
    int i = l - 1, j = r + 1;
    //划分成左右两个部分
    while(i < j){
        while(a[++i] < x);
        while(a[--j] > x);
        if(i < j){
            swap(a[i], a[j]);
        }
    }
    //对左部分排序
    quickSort(a, l, j);
    //对右部分排序
    quickSort(a, j + 1, r);
}

javascript

scss 复制代码
const quickSort = (arr,l,r) => {
    if(l >= r) return;
    let i = l - 1, j = r + 1;
    // 取分界点
    let mid = arr[l + r >> 1];
    while(i < j) {
        while(arr[++i] < mid);
        while(arr[--j] > mid);
        if(i < j) {
            [arr[i], arr[j]] = [arr[j], arr[i]];
        }
    }
    quickSort(arr,l ,j );
    quickSort(arr,j + 1, r);
}

时间复杂度

最好情况:O(nlong)

最坏情况:O(n^2)

平均时间复杂度:O(nlogn)

归并排序

归并排序的基本思想也是分治。快排的分界点是一个数,而归并排序是使用中间点来分的

  1. 确定分界点mid = (l + r) >> 1
  2. 递归排序left,right
  3. 合并两个排好序的数组

代码

c++

perl 复制代码
void merge_sort(int q[], int l, int r)
{
    if (l >= r) return; //如果区间个数是1个或者没有,return
​
    int mid = l + r >> 1;
​
    merge_sort(q, l, mid), merge_sort(q, mid + 1, r);//递归排左边和右边
​
    int k = 0, i = l, j = mid + 1;//k表示tmp里面数的个数(两个序列合并了几个数)i是左半边的起点,j是右半边的起点
    while(i <= mid && j <= r) //i<=左半边边界,j<=右半边边界
    {
        if(q[i] <= q[j]) tmp[k ++] = q[i ++];//答案把q[i]拿过来
        else tmp[k ++] = q[j ++];
    }
    while(i <= mid) tmp[k ++] = q[i ++];//把剩下的加上
    while(j <= r) tmp[k ++] = q[j ++];
​
    for(i = l, j = 0; i <= r; i ++, j ++) q[i] = tmp[j];//把存在tmp的结果复制到q里面
​
}

javascript

scss 复制代码
function merge(arr,l,r) {
    if(l >= r) return;
    const mid = l + r >> 1;
    merge(arr,l,mid);
    merge(arr, mid + 1, r)
    let i = l, j = mid + 1, k = 0, tmp = [];
    while(i <= mid && j <= r) {
        if(arr[i] <= arr[j]) {
            tmp[k++] = arr[i++];
        }else {
            tmp[k++] = arr[j++]
        }
    }
    while(i <= mid) {
        tmp[k++] = arr[i++];
    }
    while(j <= r) {
        tmp[k++] = arr[j++];
    }
    for(let i = 0; i < tmp.length; i++) {
        arr[l++] = tmp[i];
    }
}

时间复杂度

最好情况:O(nlong)

最坏情况:O(nlong)

平均时间复杂度:O(nlogn)

相关推荐
kyriewen36 分钟前
Git Commit 前自动修复代码风格?配置 Husky + lint-staged,从此 CR 只聊逻辑
前端·git·面试
小和尚同志1 小时前
AI 自动化测试探索(一):Playwright MCP
前端·人工智能·aigc
老马识途2.01 小时前
在AI的帮助下理解spring的启动过程
java·前端·spring
徐小夕2 小时前
Loop Engineering 深度解析与实战指南(全网最全)
前端·算法·github
运筹vivo@2 小时前
Python ContextVar 底层机制与内存模型拆解
前端·数据库·python
#麻辣小龙虾#3 小时前
基于vue3.0开发一款【固废与废气运维管理系统】(支持源码)
前端·vue.js·vue3
Cosolar3 小时前
Docsify零构建文档站完全指南:从快速搭建到企业级部署
前端·开源·github
weixin_471383033 小时前
Taro-02-页面路由
前端·taro
星栈独行4 小时前
Makepad 应用如何读文件、调接口、保存数据
前端·程序人生·ui·rust·github
IT_陈寒4 小时前
Vite热更新失效?可能你在用Windows
前端·人工智能·后端