时间:2021-05-20
前言
上一篇文章写了一个自顶向下的归并排序,把一个完整的数组不断二分,然后再合并。其实换一种思路:把数组中相邻的N个元素看成是已经二分好了的,直接进行合并,就省掉了二分那一步骤
C++实现:
template<typename T>void mergeSortButton2Top(T arr[], int n) { for (int size = 1; size <= n; size += size) { for (int i = 0; i+size < n; i+=2*size) //对[i,i+size-1]和[i+size,i+2*size-1]进行归并 __merge(arr, i, i + size - 1, min(i + size + size - 1,n-1));// arr left mid right 如果i+2*size>n了,越界了,就取n-1 }}template<typename T>void __merge(T arr[], int left, int mid, int right) { //将arr[left,mid] 和 arr[mid+1,right] 两部分进行归并 T *tmp=new T[right-left+1]; for (int i = left; i <= right; i++) tmp[i - left] = arr[i]; //先把arr(需要合并的左右片段) 复制给tmp int i = left, j = mid + 1; // i 做为左半部分的指针 j作为右半部分的指针 for (int k = left; k <= right; k++) { if (i > mid) { // 左半部分 已经合入完了,将右半部分剩下的 全部合入 arr[k] = tmp[j - left]; j++; } else if (j > right) { // 右半部分 已经合入完了,将左半部分剩下的 全部合入 arr[k] = tmp[i - left]; i++; } else if (tmp[i - left] < tmp[j - left]) { arr[k] = tmp[i - left]; i++; } else { arr[k] = tmp[j - left]; j++; } } delete[] tmp;}int main() { int arr[9] = { 1,5,6,78,12,5,1,12,54 }; mergeSortButton2Top(arr,9); for (int i = 0; i < 9; i++) { cout << arr[i]<<" "; } return 0;}GoLang实现:
func mergeSortButton2Top(arr [] int) { var lenth int = len(arr) for size := 1; size <= lenth; size += size { for i := 0; i+size < lenth; i += 2 * size { //对[i,i+size-1]和[i+size,i+2*size-1]进行归并 merge(arr, i, i+size-1, int(math.Min(float64(i+2*size-1), float64(lenth-1))))// arr left mid right 如果i+2*size>n了,越界了,就取n-1 } }}func merge(arr []int, left, mid, right int) { // 将要合并的部分做个拷贝 var tmp []int = make([]int, right-left+1) for i, j := left, 0; i <= right; i++ { tmp[j] = arr[i] j++ } // i做为左半部分的指针 j作为右半部分的指针 var i, j int = left, mid+1 for k := left; k <= right; k++ { if i > mid { // 左半部分 已经合入完了,将右半部分剩下的 全部合入 arr[k] = tmp[j-left] j++ } else if j > right { // 右半部分 已经合入完了,将左半部分剩下的 全部合入 arr[k] = tmp[i-left] i++ } else if tmp[i-left] > tmp[j-left] { arr[k] = tmp[j-left] j++ } else { arr[k] = tmp[i-left] i++ } }}用golang对两种归并排序进行计时,观察性能:
func createRandomArray(count int) []int { rand.Seed(time.Now().UnixNano()) var arr [] int = make([]int, 0) for i := 0; i < count; i++ { arr = append(arr, rand.Intn(100)) } return arr}func main() { count := 10000 arr := createRandomArray(count) var arr2 []int = make([]int, count) copy(arr2, arr) start := time.Now() mergeSort(arr, 0, len(arr)-1) fmt.Println("自顶向下归并排序 用时:", time.Since(start)) start = time.Now() mergeSortButton2Top(arr2) fmt.Println("自底向上归并排序 用时:", time.Since(start))}//输出://自顶向下归并排序 用时: 4.997ms//自底向上归并排序 用时: 3.9987ms因为自底向上少了二分那个步骤,性能要优于自顶向下的归并排序
总结
到此这篇关于C++/GoLang如何实现自底向上的归并排序的文章就介绍到这了,更多相关C++/GoLang自底向上的归并排序内容请搜索以前的文章或继续浏览下面的相关文章希望大家以后多多支持!
声明:本页内容来源网络,仅供用户参考;我单位不保证亦不表示资料全面及准确无误,也不保证亦不表示这些资料为最新信息,如因任何原因,本网内容或者用户因倚赖本网内容造成任何损失或损害,我单位将不会负任何法律责任。如涉及版权问题,请提交至online#300.cn邮箱联系删除。
本文实例讲述了C++实现自底向上的归并排序算法。分享给大家供大家参考,具体如下:一.算法描述自底向上的归并排序:归并排序主要是完成将若干个有序子序列合并成一个完
本文实例讲述了C++实现自顶向下的归并排序算法。分享给大家供大家参考,具体如下:一.算法描述自顶向下的归并排序:采用分治法进行自顶向下的程序设计方式,分治法的核
本文实例讲述了C++实现的归并排序算法。分享给大家供大家参考,具体如下:归并排序归并排序(MERGE-SORT)是建立在归并操作上的一种有效的排序算法。该算法是
归并排序思路:将数组不断二分,然后合并为有序数组C++实现:voidmergeSort(Tarr[],intleft,intright){//对arr[left
C语言实现归并排序算法归并排序(Mergesort)是创建在归并操作上的一种有效的排序算法。该算法是采用分治法(DivideandConquer)的一个非常典型