归并排序的思想是,将小的有序数组归并为一个大的有序数组,也就是对于一个数组,将其不管拆分为小的数组进行排序,然后将排好序的小数组再归并为稍大的数组,再将稍大的数组归并为原来的数组。
归并排序的是稳定排序,其时间复杂度为
归并的时候要判断三个条件:
1、当左半边数组用尽的时候要取右半边的数组;
2、当右半边数组用尽的时候要取左半边的数组;
3、如果未用尽,右边当前的元素与左边当前的元素作比较,取小的(升序)或者取大的(降序)。
归并算法可以有两种做法,自顶向下和自底向上。
1、自顶向下
自顶向下的做法是先拆分,再归并。也就是先将原来的数组拆分到不能再拆分为止,然后再注意归并回来,实现的时候需要有一个缓存数组。先上代码,首先需要有一个原地归并函数,就是用于判断上面提到的三个条件:
void QtSort::merge(int* pInput, int low, int mid, int high)
{
int i = low;
int j = mid + 1;
for (int k = low; k <= high; k++)
{
*(m_pAUX + k) = *(pInput + k);
}
for (int k = low; k <= high; k++)
{
if (i > mid)
{
*(pInput + k) = *(m_pAUX + (j++));
}
else if (j>high)
{
*(pInput + k) = *(m_pAUX + (i++));
}
else if (less(*(m_pAUX + j), *(m_pAUX + i)))
{
*(pInput + k) = *(m_pAUX + (j++));
}
else
{
*(pInput + k) = *(m_pAUX + (i++));
}
}
}
然后是调用递归的做法来实现排序:
void QtSort::mergeSort(int* pInput, int low, int high)
{
if (high <= low)
{
return;
}
int mid = low + (high - low) / 2;
mergeSort(pInput, low, mid);
mergeSort(pInput, mid + 1, high);
merge(pInput, low, mid, high);
}
可看到,归并第一步就是不断的拆分,拆分之后再归并回来,给一张网上找来的图像,侵删:
2、自底向上
自底向上的做法可以不用递归来做,而是跳跃着将小块排好序再对大块进行排序,其代码与自顶向下的区别主要是在mergesor函数上,其他基本一样:
void QtSort::mergeSort1(int* pInput)
{
for (int i = 1; i < dLen; i += i)
{
for (int j = 0; j < dLen - i; j += i * 2)
{
int dMin = (j + i * 2 - 1) < (dLen - 1) ? (j + i * 2 - 1) : (dLen - 1);
merge(pInput, j, j + i - 1, dMin);
}
}
}
可见,用到两个循环,第一个循环控制块的大小,第二个循环才是对小块进行归并。这样的话不会用到递归来操作,但是相对不好理解一点。一个简单的示意图推到如下,逐层将同颜色的块进行归并:
夕阳谁唤下楼梯,
一握香荑。
回头忍笑阶前立,
总无语,也依依。
笺书直恁无凭据,
休说相思。
劝伊好向红窗醉,
须莫及,落花时。