归并排序分析
admin
2024-05-02 21:24:33

目录

🥝 原理介绍:

🥝 特点:

🥝 代码示例:

🥝 算法复杂度:

🥝 改进方案:


🥝 原理介绍:

归并排序就是将两个或两个以上的有序表合并成一个有序表的过程。将两个有序表合并成一个有序表的过程称为2-路归并。

🥝 特点:

  1. 稳定性:归并排序是一种稳定的排序算法,它保证相同的元素在排序后相对位置不变。

  2. 效率:归并排序是一种较快的排序算法,在排序大型数据时,它的效率很高。

  3. 适用范围:归并排序适用于排序数据规模较大的情况,因为它的常数较小,在排序大型数据时,它的效率很高。

🥝 代码示例:

递归代码:

/***************************************************************** @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),一般来讲,基于从单个记录开始两两归并的排序并不是特别提倡,一 种比较常用的改进就是结合插入排序,即先利用插入排序获得较长的有序子序列,然后再两两归并(改进后的归并仍是稳定的,因为插入排序是稳定的)。

相关内容

热门资讯

天山冰雪迎客来(图片新闻) 天... 新疆维吾尔自治区昌吉回族自治州天山天池风景区利用丰富的冰雪旅游资源开展多场文化、体育、旅游等冰雪相关...
喝酒,要选晚上,才不会误事!暮... 喝酒,要选晚上,才不会误事! 暮色四合时小酌最为相宜。这不仅顺应人体自然节律,更蕴含着养生妙谛。晨...
吃广西横县的鸭肉生,端上桌鸭肉... 作者丨发财金刚 广西横县人对美食的追求,有自己的一套赤诚法则。 高端的食材往往只需要简单的烹饪,这句...
白茶的冲泡方法 冲泡白茶常见的方法有杯泡法、壶泡法、煮饮法、盖碗法、冷泡法等。 (1) 杯泡法。对于白毫银针和等级较...
原创 川... 标题:川菜大厨:腌腊肉时,别只会放盐,多做2步,腊味提升“1倍” 在四川的厨房里,腌制腊肉是一门艺...