详解归并排序

一、问题引入

键盘输入两个有序(从小到大)的整数数组(各四个元素),输出两个数组的所有元素,以从小到大的顺序。

样例

输入:

1 5 9 12
2 8 10 11

输出:
1 2 5 8 9 10 11 12

注意:此题算法越优化,分数越高。仅仅AC,只能得基本分60分。

二、解决

1.first_merge

注意,在这道题目中,你当然可以使用冒泡排序AC,但是有比冒泡排序更好的方法。

在这里把问题泛化,将数组长度视为可变。

上代码:

#include<bits/stdc++.h>
using namespace std;
int array[1000];
int len1,len2,len;
void first_merge(int *A1, int *A2);
void show(int *arr);
int main(){
    cin>>len1>>len2;
    len = len1 + len2;
    int A1[len1];
    int A2[len2];
    for(int i = 0;i < len1;i++) cin>>A1[i];
    for(int i = 0;i < len2;i++) cin>>A2[i];
    first_merge(A1,A2);
    show(array);
    return 0;
}
void first_merge(int *A1, int *A2){
    int cnt = 0;
    int i = 0, j = 0;
    while(i < len1 && j < len2){
        if(A1[i] < A2[j]){
            array[cnt] = A1[i];
            i++;
        }
        else{
            array[cnt] = A2[j];
            j++;
        }
        cnt++;
    }
    while(i < len1){
        array[cnt] = A1[i];
        i++;
        cnt++;
    }
    while(j < len2){
        array[cnt] = A2[j];
        j++;
        cnt++;
    }        
}
void show(int *arr){
    for(int i = 0;i < len;i++) 
    cout<<arr[i]<<" ";
}

1.1代码分析

把这种问题类比为两堆已经排好序的牌,做相应的操作使其有序。见下图:

请添加图片描述
请添加图片描述

下面是代码注释和分析:

#include<bits/stdc++.h>//万能头文件
using namespace std;
int array[1000];//最终要呈现的数组
int len1,len2,len;//len1、len2、len分别对应A1、A2、array的长度
void first_merge(int *A1, int *A2);//排序函数,参数为要排序的两个数组
void show(int *arr);//打印输出函数,接受数组为参数
int main(){
    cout<<"请输入两个数组的长度"<<endl;
    cin>>len1>>len2;
    len = len1 + len2;
    int A1[len1];//动态数组
    int A2[len2];
    cout<<"请输入要排序的两个数组"<<endl;
    for(int i = 0;i < len1;i++) cin>>A1[i];//用户输入
    for(int i = 0;i < len2;i++) cin>>A2[i];
    first_merge(A1,A2);
    show(array);
    return 0;
}
void first_merge(int *A1, int *A2){
    int cnt = 0;//cnt是一个计数器,代表了array中有的元素个数
    int i = 0, j = 0;//相当于两个标记(指针),指向A1,A2的最前端元素
    while(i < len1 && j < len2)//第一种情况:两个数组没有遍历完,还有元素没有被访问
        if(A1[i] < A2[j]){//从小往大排
            array[cnt] = A1[i];//写入array
            i++;//标记移动
        }
        else{
            array[cnt] = A2[j];
            j++;//标记移动
        }
        cnt++;//计数器加一(array中多了一个元素)
    }
//下面两种情况是至少一个数组中所有元素已经用完的情况
    while(i < len1){//A1还没有用完
        array[cnt] = A1[i];
        i++;
        cnt++;
    }
    while(j < len2){//A2还没有用完
        array[cnt] = A2[j];
        j++;
        cnt++;
    }
//A1,A2都用完了,相当于len == cnt,循环终止
}
void show(int *arr){
    //打印输出
    for(int i = 0;i < len;i++) 
    cout<<arr[i]<<" ";
}

1.2 first_merge的常见问题

1.1.1 在函数内使用sizeof(int *p)
#include<iostream>
using namespace std;
int array[1000];
int len;
void show(int *arr);
void merge(int *A1,int *A2);
int main(){
	int a[] = {1,2,5,9,14,19};
	int b[] = {3,4,6,10,15,23};
	merge(a,b);
	for(int i = 0;i < len;i++) 
	cout<<array[i]<<" ";
	return 0;
}
void merge(int *A1,int *A2){
	int len1 = sizeof(A1)/sizeof(A1[0]);//注意这一行
	int len2 = sizeof(A2)/sizeof(A2[0]);//注意这一行
	len = len1 + len2;
	int i = 0, j = 0,cnt = 0;
		while(i < len1 && j < len2){
		if(A1[i] < A2[j]){
			array[cnt] = A1[i];
			i++;
		}
		else{
			array[cnt] = A2[j];
			j++;
		}
		cnt++;
	}
	while(i < len1){
		array[cnt] = A1[i];
		i++;
		cnt++;
	}
	while(j < len2){
		array[cnt] = A2[j];
		j++;
		cnt++;
	}		
}

结果是这样的:

请添加图片描述

这是一种很常见的错误,原因是在函数内部使用了sizeof()去求length.

我们加入几行代码:

cout<<len<<endl;
cout<<len1<<endl;
cout<<len2<<endl;

请添加图片描述

在32位平台下,无论指针的类型是什么,sizeof(指针名)都是4,在64位平台下,无论指针的类型是什么,sizeof(指针名)都是8。

在本程序中,可以看到,不管是什么类型的指针,用sizeof()之后得到的都是8.

而程序的得到的2,2,4,是因为sizeof(int) = 4

8 / 4 = 2

数组名虽然可以视作指针,但对sizeof来说,sizeof(数组名)和sizeof(指针)是不同的。

不管是函数传进去的参数:

void function(int *array);

还是动态内存分配时用的:

int n;
cin>>n;
int *array = new int[n];

sizeof(int *array)得到的都是8;

!注意:只有一个数组arr的长度一开始就被指定好了,sizeof(arr)/sizeof(a[0])才是数组长度

比如:

int array = {1,2,3,4,5,};

或:

void funtion(int array[N]);

解决方法:

1.把数组长度当参数传进去

2.使用全局变量(本题解法)

1.1.2 长度确定的数组作为形参传输时

如果我们希望传参(数组是参数,也就是指针)时按照引用传递;
注意:这里的假设是十分特殊的,就是想用数组的引用
有人会这么写:

...
void function(int *&px);
...
int a[] = {1,2,5,9,14,19};
int *p = a;
function(p);
...

这是没有意义的,因为

int *p = a 这条语句实际上是做了一个指针的赋值,和a = b没有什么差别;
而int *&px = p; 实际上px已经成为p的引用,和数组没有什么关系了,我们希望的是传入数组的引用。
我们需要的是一个指针的引用,或者指向指针的指针。

修改程序为下面的样子:

...
void function(int *&px);
...
int a[] = {1,2,5,9,14,19};
int * & p = a;
function(p);
...

程序还是会提示出错;

[Error] cannot bind non-const lvalue reference of type 'int*&' to an rvalue of type 'int*'

注意这个 ‘non-const’, 数组名a,即数组的首地址是‘不变’的,在编译器眼中为const.

而你实际上声明了一个指针p来作为a的引用,p所指向的对象是可变的,因此是 ‘non-const’.

在编译器眼中,把一个‘non-const’作为’const’的引用是非法的。

改成:

...
void function(int * const &px);
...
int a[] = {1,2,5,9,14,19};
int * const & p = a;
function(p);
...

可以正常运行。

1.1.3遗漏情况
while(i < len1){
		array[cnt] = A1[i];
		i++;
		cnt++;
	}
	while(j < len2){
		array[cnt] = A2[j];
		j++;
		cnt++;
	}		

上面的代码特别容易被遗忘,而且不容易发现(因为有可能得到正确的答案)

遗忘后可能出现结果中有 0,0,0这样的数字。

三、分治

1. second_merge

分治是一种非常重要的算法思想,简单来讲就是‘分而治之’,把大的问题切割成小的问题,然后逐个解决。

在first_merge中,我们已经可以做到把两个有序的数组合并成一个新的数组。

现在的新问题是,给一个任意的数组,怎么把它变成有序?

第一步:分——把数组切割,直到每一个小数组有序。

第二步:合——用类似first_merge的方法把一个个有序的数组合并。

既然first_merge针对两个数组,我们就每次把数组切割成两份,最后合并的都是两个两个的数组。

注意:单个的数就是视作有序的!这将是递归的起点

代码:

#include<bits/stdc++.h>
using namespace std;
const int N = 1000;
int A[N];
int B[N];
void second_merge(int left, int right){
	if(left == right) return;
	int middle = (left + right) / 2;
	second_merge(left, middle);
	second_merge(middle + 1, right);
	int i = left;
	int j = middle + 1;
	int cnt = left;
	while(i <= middle && j <= right){
		if(A[i] < A[j]){
			B[cnt] = A[i];
			i++;
			cnt++;
		}else{
			B[cnt] = A[j];
			j++;
			cnt++;
		}
	}
	while(i <= middle){
		B[cnt] = A[i];
		i++;
		cnt++;
	}
	while(j <= right){
		B[cnt] = A[j];
		j++;
		cnt++;
	}
	for(int k = left;k <= right;k++){
		A[k] = B[k];
	}
}
int main(){
	int n;
	cin>>n;
	for(int i = 0;i<n;i++) cin>>A[i];
	second_merge(0,n-1);
	for(int i = 0;i<n;i++) cout<<A[i]<<" ";
	return 0;
}

1.1注释和分析

#include<bits/stdc++.h>
using namespace std;
const int N = 1000;//指定好长度 
int A[N];
int B[N];
void second_merge(int left, int right){
	if(left == right) return;//递归的终止条件 
	int middle = (left + right) / 2;//分割数组 
	second_merge(left, middle);// 搜索左边 
	second_merge(middle + 1, right);//搜索右边 
	int i = left;//定义两个指针:i 
	int j = middle + 1;//指针j
	int cnt = left;//记录初始位置 
	//下面是一个类似的merge
	while(i <= middle && j <= right){
		if(A[i] < A[j]){
			B[cnt] = A[i];
			i++;
			cnt++;
		}else{
			B[cnt] = A[j];
			j++;
			cnt++;
		}
	}
	while(i <= middle){
		B[cnt] = A[i];
		i++;
		cnt++;
	}
	while(j <= right){
		B[cnt] = A[j];
		j++;
		cnt++;
	}
	//下边将把B数组录入A中 
	for(int k = left;k <= right;k++){
		A[k] = B[k];
	}
}
int main(){
	int n;
	cin>>n;
	for(int i = 0;i<n;i++) cin>>A[i];
	second_merge(0,n-1);//设定初始的值 
	for(int i = 0;i<n;i++) cout<<A[i]<<" ";
	return 0;
}

2.和first_merge的区别与联系

看下面两段代码:

一、

	int i = 0, j = 0,cnt = 0;
	while(i < len1 && j < len2){
		if(A1[i] < A2[j]){
			array[cnt] = A1[i];
			i++;
		}
		else{
			array[cnt] = A2[j];
			j++;
		}
		cnt++;
	}
	while(i < len1){
		array[cnt] = A1[i];
		i++;
		cnt++;
	}
	while(j < len2){
		array[cnt] = A2[j];
		j++;
		cnt++;
	}

二、

	int i = left; 
	int j = middle + 1;
	int cnt = left;
	while(i <= middle && j <= right){
		if(A[i] < A[j]){
			B[cnt] = A[i];
			i++;
			cnt++;
		}else{
			B[cnt] = A[j];
			j++;
			cnt++;
		}
	}
	while(i <= middle){
		B[cnt] = A[i];
		i++;
		cnt++;
	}
	while(j <= right){
		B[cnt] = A[j];
		j++;
		cnt++;
	}

可以看到,两个算法的核心是及其相似的,分治后做类似的first_merge,然后合并、递归,得到对一个数组的归并排序方法。

区别:

区别1.first_merge中,我们使用了i,j两个指针,初始值赋为0,然后比较i与len1,j与len2;

second_merge中,我们使用i,j两个指针,初始值赋为left、middle + 1,然后比较i,j与middle,right.

区别2.first_merge一次就排序完毕,而second_merge中运用了递归,相当与把first_merge做了多次。

区别3.second_merge开辟了一个新的数组B。

3.注意问题和容易出错之处

3.1一些细节问题的个人理解

3.1.1关于两个指针

在first_merge中,i,j两个指针初始化为0,是因为对A1、A2来讲,指针指向的初始位置就是A1和A2头部。

second_merge中,切割后的每个小数组的头部是它对应的left。

因此对一个数组A来说,做一次切割分为两个数组后,有:

请添加图片描述
请添加图片描述

3.1.2关于等于号

请添加图片描述
解决:
请添加图片描述

可以发现,两者是等价的。

四、简化的最终版

#include <iostream>

using std::cin;
using std::cout;
using std::cerr;

const int N = 1e5 + 7;

int a[N],b[N];

void sort (int L,int R) {
	if (L == R) return;
	int M = (L + R) / 2;
	sort(L, M); sort(M + 1, R);
	int l = L, r = M + 1, t = L;
	while (l <= M && r <= R)
		if (a[l] <= a[r])
			b[t++] = a[l++];
		else 
			b[t++] = a[r++];
	while (l <= M)
		b[t++] = a[l++];
	while (r <= R)
		b[t++] = a[r++];
	for (int i = L; i <= R; ++i)
		a[i] = b[i];
}

int n;
int main(){	
	cin >> n;
	for (int i = 1; i <= n; ++i)
		cin >> a[i];
	sort(1, n);
	for (int i = 1; i <= n; ++i)
		cout << a[i] <<" ";
	return 0;
}

文章定有不当之处,还请方家批评指正。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值