归并排序的思想上我们已经全部介绍完了,但是同时也面临和快速排序一样的问题那就是递归消耗的栈帧空间太大了,所以对此我们必须掌握非递归的排序思想。
归并实现的思想无非就是先将 每个数都递归 分割为一个小区间然后再进行排序,之后递归 回溯 上一个区间 这时 上一个区间都排好了所以可以在进行排序就这样循环上去。
既然要用非递归那么我们是不是可以这样想 直接吧每个区间定义为 1 进行归并然后再来进行循环到上一组归并排序:
这样就可以利用循环来吧归并排序非递归化了
好了具体思想那么我们懂了,既然要进行从最小区间 1 开始那么我们肯定需要需要定义 一个 gap = 1 开始循环
gap * = 2;
来进行调整我们的归并区间的间距进行归并int i = 0; i < n; i += (gap*2)
i += (gap*2)
因为 每次当这个区间排完了之后就需要跳到下一个区间开始代码演示:
// 归并排序非递归实现
void MergeSortNonR(int* a, int n)
{
int* tmp = (int*)malloc(sizeof(int) * n);
if (tmp == NULL)
{
perror("malloc file");
exit(-1);
}
int gap = 1;
while (gap < n)
{
for (int i = 0; i < n; i += (gap*2))
{
int begin1 = i, end1 = i + gap - 1;
int begin2 = i + gap, end2 = i + (2 * gap) - 1;
int index = i;
while (begin1 <= end1 && begin2 <= end2)
{
if (a[begin1] < a[begin2])
{
tmp[index++] = a[begin1++];
}
else
{
tmp[index++] = a[begin2++];
}
}
while (begin1 <= end1)
{
tmp[index++] = a[begin1++];
}
while (begin2 <= end2)
{
tmp[index++] = a[begin2++];
}
memcpy(a + i, tmp + i, sizeof(int) * (2*gap));
}
printf("\n");
gap *= 2;
}
free(tmp);
}
以上就是非递归实现的代码了,但你真的以为非递归就这样结束了?哈哈哈其实没有我们前面举例的是2的倍数来进行排序的但是当我们排序10之类的不是2的倍数就会出现越界的情况:
注:是上面我们每次 第二个区间都是 i + (2 * gap) - 1 但是当不是2的整数倍来实现的话不就越界了
哦豁大家是不是看到了当第二次排序的时候 begin2
和 end2
都越界了,第三次归并的时候甚至 end2
都 越界了
这其实非常简单既然第二个区间都越界的话那么是不是就不需要进行归并了,你想啊连第二个区间都不存在的话第一个区间和谁归并?
注:这里还要注意memcpy 的时候的copy大小。
以前我们进行 copy 的时候都是 2倍的gap 但是当才不是整数倍的时候就需要调整了
i 每次都是要归并的区间开头, 而 end2 倍修正了之后就是区间尾了他们一相减就好了
注:相减了之后要加1,因为是闭区间。(3-0)虽然是相减了但是我们实际复制的是4个数
// 归并排序非递归实现
void MergeSortNonR(int* a, int n)
{
int* tmp = (int*)malloc(sizeof(int) * n);
if (tmp == NULL)
{
perror("malloc file");
exit(-1);
}
int gap = 1;
while (gap < n)
{
for (int i = 0; i < n; i += (gap*2))
{
int begin1 = i, end1 = i + gap - 1;
int begin2 = i + gap, end2 = i + (2 * gap) - 1;
if (begin2 >= n)
break;
if (end2 >= n)
end2 = n-1;
printf("[%d,%d][%d,%d] ", begin1, end1, begin2, end2);
int index = i;
while (begin1 <= end1 && begin2 <= end2)
{
if (a[begin1] < a[begin2])
{
tmp[index++] = a[begin1++];
}
else
{
tmp[index++] = a[begin2++];
}
}
while (begin1 <= end1)
{
tmp[index++] = a[begin1++];
}
while (begin2 <= end2)
{
tmp[index++] = a[begin2++];
}
memcpy(a + i, tmp + i, sizeof(int) * (end2-i+1));
}
printf("\n");
gap *= 2;
}
free(tmp);
}
看到这里了还不给博主扣个:
⛳️ 点赞
收藏
⭐️ 关注
!
❤️
拜托拜托这个真的很重要!
你们的点赞就是博主更新最大的动力!
有问题可以评论或者私信呢秒回哦。