zoukankan      html  css  js  c++  java
  • 排序之归并排序

    归并排序的思想是,将小的有序数组归并为一个大的有序数组,也就是对于一个数组,将其不管拆分为小的数组进行排序,然后将排好序的小数组再归并为稍大的数组,再将稍大的数组归并为原来的数组。

    归并排序的是稳定排序,其时间复杂度为NlogN

    归并的时候要判断三个条件:

    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);
    	    }
    	}
    }

    可见,用到两个循环,第一个循环控制块的大小,第二个循环才是对小块进行归并。这样的话不会用到递归来操作,但是相对不好理解一点。一个简单的示意图推到如下,逐层将同颜色的块进行归并:

    夕阳谁唤下楼梯,

    一握香荑。

    回头忍笑阶前立,

    总无语,也依依。

    笺书直恁无凭据,

    休说相思。

    劝伊好向红窗醉,

    须莫及,落花时。

    上善若水,为而不争。
  • 相关阅读:
    sqlserver判断字段是否存在,表是否存在
    sqlserver数据库数据字典生成器
    C#断点续传下载文件
    c# 泛型new T
    html显示xml内容
    程序设计语言诞生——程序设计语言的发展历史
    atan2(x,y) pow(x,y)
    name phone email 正则表达式
    第八周
    第六周
  • 原文地址:https://www.cnblogs.com/Bearoom/p/11721783.html
Copyright © 2011-2022 走看看