找回密码
 注册
关于网站域名变更的通知
查看: 341|回复: 1
打印 上一主题 下一主题

2020-1024=996:归并排序!

[复制链接]

该用户从未签到

跳转到指定楼层
1#
发表于 2020-10-26 10:53 | 只看该作者 |只看大图 回帖奖励 |倒序浏览 |阅读模式

EDA365欢迎您登录!

您需要 登录 才可以下载或查看,没有帐号?注册

x
今天给大家分享排序算法里面的另外一种排序算法:归并排序!0 T8 }- Z/ T3 L% L* Y: g' ~

. O; D9 P) m9 O# r* _6 Q* |
( K4 r6 W! V4 ]$ {+ S

0 g- A  @9 e# C4 h) i% p

一、归并排序:

1、归并排序操作的核心思想:

a、确定分界点:mid=(l+r)/2

b、递归排序左边和右边(排完左右两边的数,就会成为两个有序的序列了)

c、归并(把上面的两个有序序列合并成一个有序的序列,用一个简单的词来说,就是合二为一!)


# ?; B* `3 v. S5 k3 ~6 z

2、举例:

比如上图我们有两组已经排好的序列数字,我们要进行第三步合并,该如何进行呢?思路如下:

a、这里先定义一个空的数组res,它主要是为了临时存放合并序列排序好的数字;我们从图中可以看到,第一个序列指针i指向数字1,第二序列指针j指向2,这个时候我们要比较两个数字的大小,小的数字就放到临时数组res里面去,这里我们明显知道数字1小于2,所以把1放到临时数组res里去


0 Y! l' j9 {" Z& W) m( j

b、然后指针i往下移动,如下图所示,再次进行比较,明显发现指针j指向的数字2更小,把它放到res里面去,然后指针j往下移动,指针i不动,后面依次类推

c、如下图所示,两个指针都指向了数字5,如果遇到两个数字一样的话,一般是把第一个序列的数字放到临时数组res里面去,这点稍微要注意一下

d、最后把临时数组里面的是数字放到原来的数组里面去

注意:一个算法稳定,并不能说它的时间效率是稳定的;这里的稳定是说两个序列中有两个数是相同的,如果在排完序之后,他们的位置还是没有发生变化的话,那么这个排序就是稳定的,反之亦然!

3、归并排序的平均时间复杂度的计算推导:

从图片的纵性来分析,当拆解到1的时候,这个时候什么数等于n除于它等于1,通过计算,我们知道是logn,然后再从横向分析,我们要最多比较n个数字,所以归并排序的时间复杂度就是:nlogn

二、代码示例:

代码:

#include <iostream>% y# }* T0 k8 }: u
using namespace std;% s8 y. @* K& C: t
const int N = 1e5 + 10;
7 L% J3 m1 K' x3 L4 Kint n;; V* F& g" o! t* {+ W! k+ w: q* g
int q[N], tmp[N];+ H6 ?7 w, S6 B5 w0 Y0 k/ J5 u
5 k! w! M# H9 M% c' Q, z+ P7 Q" c
void merge_sort(int q[],int l, int r)$ D% |* I  [6 B' n9 j% \
{
5 w9 ]: ^' Z4 V if(l>=r)return;//判断序列中是否为空或者只有一个数字,如果是的话,我们就不用排序了7 f- _) ~0 k* E% }2 V
//确定分界点! b% p: f; j, p( p" C: f; m
int mid = l + r >> 1;
5 E- M8 ~  v5 |& ~4 U //递归处理# g$ y$ p1 H4 @+ M: a% ?" \
merge_sort(q,l,mid);
0 }  q2 [* P- V% K: X6 G merge_sort(q,mid+1,r);) h+ G) s( Z8 W( l/ \8 c! Z
//定义双指针. L7 O6 O- H( H
int k =0,i = l, j= mid+1;" F$ e; s: {  }' p' \
//归并处理0 P" P$ W8 Y" Y
while(i <= mid && j <= r)
/ ~% ], u  B- q0 @% b  if(q < q[j])tmp[k++] = q[i++];+ e4 l5 j3 p0 W
else
6 l- z- ~$ \( q  tmp[k++] = q[j++];
1 [) Y3 w; {* v: X //把源数组中剩余的数字(注意这里的数字一定是最小的)放到临时数组后面去
( k* O8 d" d) @- S while(i <= mid)tmp[k++] = q[i++];
0 C  ^0 |) B) ]) \# c" W3 l3 u3 c while(j <= r)tmp[k++] = q[j++];
8 m) ~$ b# h. Y' r/ @, l% W4 @ //把临时数组中排好序的数字放到源数组中去
, P: L; G, t' u/ O9 j( U for(i = l, j =0;i<=r;i++,j++)q=tmp[j];) i. r0 J- a' c9 x) I
}0 x: @' A. R. j! Q

# o& B8 u: w7 u) ^3 H- b& T
% r/ A  j' a7 Q" t9 K. Vint main(). z1 N# x# A4 x' A6 V# c0 {& ?. ^
{
" ?6 M, B! d) r9 x+ W# _ scanf("%d",&n);
$ P  [/ H7 g3 f; _; n; j for(int i = 0;i<n;i++)3 ^, y: ?& Z( L5 z5 Q& T
{9 C# F( b0 \# u
  scanf("%d",&q);
4 o& C) f: M: I. u1 X0 O; I }
4 a7 ]+ v" o5 |+ L) V1 a+ g merge_sort(q,0,n-1);' k: H- S5 n0 B+ F! e' s
for(int i = 0; i<n;i++)6 c. }! q2 u* @2 ?" Z' \  O
{
: F( l! r* H( p  printf("%d ",q);2 L, n2 ?% j/ v* S
}+ C: _4 A- {# p$ p, z
return 0;
8 E. Y# ^3 m* ?% Q
* l7 N2 l# p: o7 p  \9 r9 ]5 _ ; A$ U# ]0 N# I; o1 M8 N* N8 L
}

4 |3 ]8 M4 C# t1 T# V, {" {
  • TA的每日心情
    开心
    2022-12-5 15:37
  • 签到天数: 2 天

    [LV.1]初来乍到

    2#
    发表于 2020-10-26 13:12 | 只看该作者
    2020-1024=996,哈哈
    您需要登录后才可以回帖 登录 | 注册

    本版积分规则

    关闭

    推荐内容上一条 /1 下一条

    EDA365公众号

    关于我们|手机版|EDA365电子论坛网 ( 粤ICP备18020198号-1 )

    GMT+8, 2025-11-24 23:27 , Processed in 0.171875 second(s), 26 queries , Gzip On.

    深圳市墨知创新科技有限公司

    地址:深圳市南山区科技生态园2栋A座805 电话:19926409050

    快速回复 返回顶部 返回列表