網(wǎng)站首頁 編程語言 正文
前言
本期為大家?guī)淼氖浅R娕判蛩惴ㄖ械?strong>歸并排序,博主在這里先分享歸并排序的遞歸算法,包您一看就會(huì),快來試試吧~
?一、歸并排序
1.1 基本思想
歸并排序(MERGE-SORT)是建立在歸并操作上的一種有效的排序算法,該算法是采用分治法 (Divide and Conquer)的一個(gè)非常典型的應(yīng)用。將已有序的子序列合并,得到完全有序的序 列;即先使每個(gè)子序列有序,再使子序列段間有序。若將兩個(gè)有序表合并成一個(gè)有序表,稱為二路歸并。
1.2 算法思想
到這里,我們可以得到一條結(jié)論,兩個(gè)有序的子序列可以合成一個(gè)新的有序子序列,通過遞歸,我們兩個(gè)新的有序序列也可以組成新的有序數(shù)列,最終實(shí)現(xiàn)排序的目的。有些朋友就會(huì)問了,這個(gè)我懂,關(guān)鍵是咋實(shí)現(xiàn)有序數(shù)列,其實(shí)非常的簡單,分割遞歸至每個(gè)子序列只有一個(gè)元素時(shí),是不是就有序啦,然后就可以合成有兩個(gè)元素有序的列表嘛,再合成4個(gè),8個(gè)……
1.3 程序設(shè)計(jì)思想
定義一個(gè)tmp數(shù)組,可以是動(dòng)態(tài)開辟的(malloc),用于臨時(shí)存放合并后的有序數(shù)據(jù),定義_MergeSort()函數(shù),用于分解,合并數(shù)據(jù)(遞歸實(shí)現(xiàn)),參數(shù)有,待處理數(shù)組,數(shù)據(jù)區(qū)間(數(shù)組下標(biāo)),tmp數(shù)組。
- 判斷區(qū)間是否存在,區(qū)間不存在以及只有一個(gè)元素的情況結(jié)束程序。
- 分割區(qū)間:mid=(left+right)/2;遞歸左右區(qū)間,分割遞歸至每個(gè)子序列只有一個(gè)元素。
- 每兩個(gè)子序列一組,循環(huán)遍歷每一個(gè)元素,if比較兩個(gè)子序列各元素的大小,取大或取小,放入tmp數(shù)組,tmp[index++]=子序列++;直到有一個(gè)子序列遍歷完,循環(huán)結(jié)束。
- 循環(huán)判斷是子序列是否遍歷完畢,未遍歷完畢的子序列剩余元素直接給到tmp數(shù)組。將tmp數(shù)組的對(duì)應(yīng)的元素拷貝回原數(shù)組(已有序)。
1.4 程序?qū)崿F(xiàn)
#define _CRT_SECURE_NO_WARNINGS
#include<stdio.h>
#include<stdlib.h>//動(dòng)態(tài)開辟空間的函數(shù)的頭文件
void _MergeSort(int *a,int left,int right,int *tmp)
{
//區(qū)間不存在以及只有一個(gè)元素的情況結(jié)束程序
if (left>=right)
{
return;
}
int mid = (left + right) / 2;
//假設(shè)[left,mid],[mid+1,right]有序,那么我們就可以歸并了
//遞歸使左右區(qū)間有序
//分割遞歸至每個(gè)子序列只有一個(gè)元素
_MergeSort(a,left,mid,tmp);
_MergeSort(a, mid+1,right, tmp);
//歸并
int begin1 = left, end1 = mid;
int begin2 = mid + 1, end2 = right;
int index = left;
while (begin1<=end1&&begin2<=end2)//有一個(gè)子序列遍歷完,循環(huán)結(jié)束
{
if (a[begin1] < a[begin2])//升序,取小
{
tmp[index++] = a[begin1++];
}
else
{
tmp[index++] = a[begin2++];
}
}
//判斷子序列是否遍歷完,未遍歷完畢的子序列剩余元素直接給到tmp數(shù)組
while (begin1 <= end1)
{
tmp[index++] = a[begin1++];
}
while (begin2<=end2)
{
tmp[index++] = a[begin2++];
}
//拷貝回去
for (int i=left;i<=right;++i)
{
a[i] = tmp[i];
}
}
//歸并排序
void MergeSort(int *a,int n)
{
int* tmp=(int*)malloc(sizeof(int)*n);//動(dòng)態(tài)開辟與待排序數(shù)組大小相等的一片連續(xù)的空間
_MergeSort(a,0,n-1,tmp);//子函數(shù)實(shí)現(xiàn)歸并
free(tmp);//釋放動(dòng)態(tài)開辟的空間
}
//打印
void Print(int* a, int n)
{
for (int i=0;i<n;++i)
{
printf("%d ",a[i]);
}
}
int main()
{
int a[] = {10,6,7,1,3,9,4,2};
MergeSort(a,sizeof(a)/sizeof(a[0]));
Print(a,sizeof(a)/sizeof(a[0]));
return 0;
}
1.5 歸并排序的特性總結(jié)
- 1. 歸并的缺點(diǎn)在于需要O(N)的空間復(fù)雜度,歸并排序的思考更多的是解決在磁盤中的外排序問 題。
- 2. 時(shí)間復(fù)雜度:O(N*logN)
- 3. 空間復(fù)雜度:O(N)
- 4. 穩(wěn)定性:穩(wěn)定
原文鏈接:https://blog.csdn.net/weixin_67603503/article/details/125718908
相關(guān)推薦
- 2022-04-12 python入門之Scrapy?shell的使用_python
- 2024-03-09 【Redis】Redis中的布隆過濾器
- 2023-01-31 MongoDB?入門指南_MongoDB
- 2022-07-15 C++面向?qū)ο笾惡蛯?duì)象那些你不知道的細(xì)節(jié)原理詳解_C 語言
- 2023-03-01 Shell?$[]對(duì)整數(shù)進(jìn)行數(shù)學(xué)運(yùn)算實(shí)現(xiàn)_linux shell
- 2022-12-01 docker?容器網(wǎng)絡(luò)模式詳解_docker
- 2022-11-10 Kotlin?協(xié)程異步熱數(shù)據(jù)流的設(shè)計(jì)與使用講解_Android
- 2023-03-26 使用docker安裝hadoop的實(shí)現(xiàn)過程_docker
- 最近更新
-
- window11 系統(tǒng)安裝 yarn
- 超詳細(xì)win安裝深度學(xué)習(xí)環(huán)境2025年最新版(
- Linux 中運(yùn)行的top命令 怎么退出?
- MySQL 中decimal 的用法? 存儲(chǔ)小
- get 、set 、toString 方法的使
- @Resource和 @Autowired注解
- Java基礎(chǔ)操作-- 運(yùn)算符,流程控制 Flo
- 1. Int 和Integer 的區(qū)別,Jav
- spring @retryable不生效的一種
- Spring Security之認(rèn)證信息的處理
- Spring Security之認(rèn)證過濾器
- Spring Security概述快速入門
- Spring Security之配置體系
- 【SpringBoot】SpringCache
- Spring Security之基于方法配置權(quán)
- redisson分布式鎖中waittime的設(shè)
- maven:解決release錯(cuò)誤:Artif
- restTemplate使用總結(jié)
- Spring Security之安全異常處理
- MybatisPlus優(yōu)雅實(shí)現(xiàn)加密?
- Spring ioc容器與Bean的生命周期。
- 【探索SpringCloud】服務(wù)發(fā)現(xiàn)-Nac
- Spring Security之基于HttpR
- Redis 底層數(shù)據(jù)結(jié)構(gòu)-簡單動(dòng)態(tài)字符串(SD
- arthas操作spring被代理目標(biāo)對(duì)象命令
- Spring中的單例模式應(yīng)用詳解
- 聊聊消息隊(duì)列,發(fā)送消息的4種方式
- bootspring第三方資源配置管理
- GIT同步修改后的遠(yuǎn)程分支