目录
🥝 原理介绍:
🥝 特点:
🥝 代码示例:
🥝 算法复杂度:
🥝 改进方案:
归并排序就是将两个或两个以上的有序表合并成一个有序表的过程。将两个有序表合并成一个有序表的过程称为2-路归并。
稳定性:归并排序是一种稳定的排序算法,它保证相同的元素在排序后相对位置不变。
效率:归并排序是一种较快的排序算法,在排序大型数据时,它的效率很高。
适用范围:归并排序适用于排序数据规模较大的情况,因为它的常数较小,在排序大型数据时,它的效率很高。
递归代码:
/***************************************************************** @description: merge sort* @author: kashine* @version: 1.0* @mail: likaiqinchina@gmail.com* @date: 2022/12/29
*/
#include
#include using namespace std;void merge(vector& array, int start, int mid, int end)
{// [start, mid]左侧有序, [mid + 1, end]右侧有序int n1 = mid - start + 1;// 左侧有序int n2 = end - mid;// 右侧有序// 将左右侧有序部分分别存储到temp1,temp2中vector temp1, temp2;int i = 0, j = 0;while(i < n1)temp1.emplace_back(array[start + i++]);while(j < n2)temp2.emplace_back(array[mid + 1 + j++]);// 将temp1,temp2按顺序归并到array中i = 0; j = 0;int k = start;// 注意:k不能等于0!!!while(i < n1 && j < n2){if(temp1[i] < temp2[j]){array[k++] = temp1[i++];}else{array[k++] = temp2[j++];}}// 将剩余元素拷贝到array中while(i < n1)array[k++] = temp1[i++];while(j < n2)array[k++] = temp2[j++];}// 归并排序
// start为排序起始下标,end为排序终止下标
// 二路归并实现排序,分治法(左、右)
void merge_sort(vector& array, int start, int end)
{if(start < end)// 如果start大于或者等于end,不进行处理,即当数组只有一个元素不做处理{int mid = (start + end) / 2;merge_sort(array, start, mid);merge_sort(array, mid + 1, end);merge(array, start, mid, end);}
}int main()
{vector array{5, 2, 4, 6, 1, 3};merge_sort(array, 0, array.size() - 1);for(auto& it : array){cout<< it << " ";}cout<
非递归代码:
/***************************************************************** @description: merge sort* @author: kashine* @version: 1.0* @mail: likaiqinchina@gmail.com* @date: 2022/12/29
****************************************************************/
#include
#include using namespace std;void merge(vector& array, int start, int mid, int end)
{// [start, mid]左侧有序, [mid + 1, end]右侧有序int n1 = mid - start + 1;// 左侧有序int n2 = end - mid;// 右侧有序// 将左右侧有序部分分别存储到temp1,temp2中vector temp1, temp2;int i = 0, j = 0;while(i < n1)temp1.emplace_back(array[start + i++]);while(j < n2)temp2.emplace_back(array[mid + 1 + j++]);// 将temp1,temp2按顺序归并到array中i = 0; j = 0;int k = start;// 注意:k不能等于0!!!while(i < n1 && j < n2){if(temp1[i] < temp2[j]){array[k++] = temp1[i++];}else{array[k++] = temp2[j++];}}// 将剩余元素拷贝到array中while(i < n1)array[k++] = temp1[i++];while(j < n2)array[k++] = temp2[j++];}void mergeSort(vector& arr) {int n = arr.size();for (int curr_size = 1; curr_size <= n - 1; curr_size = 2 * curr_size) {for (int left_start = 0; left_start < n - 1; left_start += 2 * curr_size) {int mid = left_start + curr_size - 1;int right_end = std::min(left_start + 2 * curr_size - 1, n - 1);merge(arr, left_start, mid, right_end);}}
}int main() {vector arr = {38, 27, 43, 3, 9, 82, 10};mergeSort(arr);for (int i = 0; i < arr.size(); i++)std::cout << arr[i] << " ";std::cout << std::endl;return 0;
}
合并复杂度O(n),分治复杂度:T(n) = 2T(n/2) + O(n),类似于快速排序每一次取到的元素都刚好平分整个数组的情况。详细推导见:快速排序时间复杂度分析 - 知乎排序算法之 快速排序 及其时间复杂度和空间复杂度
最好时间复杂度:O(nlog2n)
最坏时间复杂度:O(nlog2n)
空间复杂度:O(n)
因为不管元素在什么情况下都要做这些步骤,所以花销的时间是不变的,所以该算法的最优时间复杂度和最差时间复杂度及平均时间复杂度都是一样的为:O( nlogn )。
详见排序算法之 归并排序 及其时间复杂度和空间复杂度
归并排序的时间复杂度为O(nlgn),一般来讲,基于从单个记录开始两两归并的排序并不是特别提倡,一 种比较常用的改进就是结合插入排序,即先利用插入排序获得较长的有序子序列,然后再两两归并(改进后的归并仍是稳定的,因为插入排序是稳定的)。
上一篇:人脸识别与美颜算法实战-图像特效
下一篇:C51 串口函数