body, table{font-family: 微软雅黑; font-size: 13.5pt}
table{border-collapse: collapse; border: solid gray; border-width: 2px 0 2px 0;}
th{border: 1px solid gray; padding: 4px; background-color: #DDD;}
td{border: 1px solid gray; padding: 4px;}
tr:nth-child(2n){background-color: #f8f8f8;}

归并排序(Merging Sort):归并排序是建立在归并操作上的一种有效的排序算法。该算法是采用分治法(Divide and Conquer)的一个非常典型的应用。
  原理:假设初始序列含有n个记录,则可以看成是n个有序的子序列,每个子序列的长度为1,然后两两归并,得到n除以2向上取整个长度为2或1的有序子序列;两两归并,直到得到一个长度为n的有序序列为止。
#include<iostream>
using namespace std;
//归并排序,先数组细分为单个,在合并
int mergeArr(int* arr,int first,int mid,int last);
int mergeSort(int* arr,int first,int last);  //递归实现
int mergeSortIterator(int* arr,int first,int last);  //非递归实现
void swap(int& elem1,int& elem2);
void test();
void printArr(int* arr,int length);
int mergeArr(int* arr,int first,int mid,int last)
{
        if(NULL==arr||first>last||first>mid||mid>last||first<0||mid<0||last<0)
                return -1;
        int len = last - first +1;
        int* tmpArr = new int[len]();
        int firIdx = first,lasIdx = mid + 1;
        int idx = 0;
        while(firIdx<=mid&&lasIdx<=last)
        {
                if(arr[firIdx]<arr[lasIdx])
                        tmpArr[idx++] = arr[firIdx++];
                else
                        tmpArr[idx++] = arr[lasIdx++];
        }
        while(firIdx<=mid)
                tmpArr[idx++] = arr[firIdx++];
        while(lasIdx<=last)
                tmpArr[idx++] = arr[lasIdx++];
        for(int iidx=first,idx=0;iidx<=last;++iidx)
                arr[iidx] = tmpArr[idx++];
        delete []tmpArr;
        return 0;
}
int mergeSort(int* arr,int first,int last)
{
        if(NULL==arr||first<0||last<0||first>last)
                return -1;
        int mid = (first + last)/2;
        if(first<last)  //划分为只有一个元素
        {
                mergeSort(arr,first,mid);
                mergeSort(arr,mid+1,last);
        }
        //合并
        mergeArr(arr,first,mid,last);
        return 0;
}
int mergeSortIterator(int* arr,int first,int last)
{
        if(NULL==arr||first<0||last<0||first>last)
                return -1;
        //非递归,直接合并
        for(int idx=1;idx<=last;idx*=2)  //都是两两合并,所以乘以2,最初从1开始,表示数组都划分为1的单个数组
        {
                int firstIdx = 0;
                while(firstIdx+idx<=last)
                {
                        int mid = firstIdx + idx -1;
                        //last有特殊情况,比如数组奇数个,最后会剩下一个元素
                        int lastIdx = mid + idx <= last ? mid + idx : last;
                        mergeArr(arr,firstIdx,mid,lastIdx);
                        firstIdx = lastIdx + 1;  //开始合并下一个子序列
                }
        }
        return 0;
}
void printArr(int* arr,int length)
{
        if(NULL==arr||length<=0)
                return ;
        for(int idx=0;idx!=length;++idx)
        {
                cout<<arr[idx]<<" ";
        }
        cout<<endl;
}
void test()
{
        int arr[] = {6,5,3,1,8,7,2,4};
        printArr(arr,8);
        mergeSort(arr,0,7);
        /*mergeSortIterator(arr,0,7);*/
        printArr(arr,8);
        cout<<endl;
        int arr1[] = {1,2,0,-1,5,6,8,3};
        printArr(arr1,8);
        mergeSort(arr1,0,7);
        /*mergeSortIterator(arr1,0,7);*/
        printArr(arr1,8);
        cout<<endl;
        int arr2[] = {2,2,2,2};
        printArr(arr2,4);
        mergeSort(arr2,0,3);
        //mergeSortIterator(arr2,0,3);
        printArr(arr2,4);
        cout<<endl;
        int arr3[] = {2,4,1};
        printArr(arr3,3);
        mergeSort(arr3,0,2);
        /*mergeSortIterator(arr3,0,2);*/
        printArr(arr3,3);
        cout<<endl;
        int arr5[] = {1,2,3,4,5,6,7,8};
        printArr(arr5,8);
        mergeSort(arr5,0,7);
        /*mergeSortIterator(arr5,0,7);*/
        printArr(arr5,8);
        cout<<endl;
        int* arr6 = NULL;
        printArr(arr6,4);
        mergeSort(arr6,0,3);
        /*mergeSortIterator(arr6,0,3);*/
        printArr(arr6,4);
        cout<<endl;
}
int main()
{
        test();
        system("pause");
}

//归并排序,原地归并,空间复杂度O(1)
//归并排序,一般的空间复杂度都是O(n),现在要O(1)
#include<iostream>
using namespace std;
void swap(int& a,int& b)
{
        int tmp = a;
        a = b;
        b = tmp;
        return ;
}
void reverse(int* arr,int len)  //len个元素的数组逆序
{
        if(NULL==arr)
                return ;
        int first = 0,end = len-1;
        while(first<end)
        {
                swap(arr[first++],arr[end--]);
        }
        return ;
}
void printArr(int* arr,int len)
{
        if(NULL==arr)
                return ;
        for(int idx=0;idx!=len;++idx)
        {
                cout<<arr[idx]<<" ";
        }
        cout<<endl;
        return;
}
void moveLeft(int* arr,int len,int cnt) //arr数组,长度len,左移cnt个元素
{
        if(NULL==arr||len<=0)
                return ;
        reverse(arr,cnt);    // 1 2 3 4 5 ,左移3个位置,arr,5,3
        reverse(arr+cnt,len-cnt);
        reverse(arr,len);
}
void mergeArr(int* arr,int first,int mid,int end)
{
        if(NULL==arr||first>end||mid>end)
                return ;
        int second = mid + 1;
        while(first<=mid&&second<=end)
        {
                int secondLow = 0;
                while( first<=mid && arr[first]<=arr[second] )
                        ++first;
                while( second<=end && arr[second]<=arr[first] )
                {
                        ++second;
                        ++secondLow;
                }
                moveLeft(arr+first,second-first,second - first - secondLow);
                first += secondLow;
        }
        return ;
}
void mergeSort(int* arr,int first,int end)
{
        if(NULL==arr)
                return;
        if(first<end)
        {
                int mid = (first + end)/2;
                mergeSort(arr,first,mid);
                mergeSort(arr,mid+1,end);
                //归并
                mergeArr(arr,first,mid,end);
        }
}
int main(int argc,char** argv)
{
        int arr[] = {6,4,3,1,7,8,2,9,5,0};
        mergeSort(arr,0,9);
        printArr(arr,10);
        system("pause");
}

归并排序(Merging Sort)的更多相关文章

  1. 数据结构 - 归并排序(merging sort)

    归并排序(merging sort): 包含2-路归并排序, 把数组拆分成两段, 使用递归, 将两个有序表合成一个新的有序表. 归并排序(merge sort)的时间复杂度是O(nlogn), 实际效 ...

  2. 数据结构 - 归并排序(merging sort) 具体解释 及 代码

    归并排序(merging sort) 具体解释 及 代码 本文地址: http://blog.csdn.net/caroline_wendy 归并排序(merging sort): 包括2-路归并排序 ...

  3. 小小c#算法题 - 8 - 归并排序 (Merging Sort)

    “归并”的含义是将两个或两个以上的有序序列组合成一个新的有序序列.这个“归并”可以在O(n+m)的数量级上实现,但这同时也需要O(n+m)的空间复杂度.具体为:首先分配一个新的长度为n+m的空序列,然 ...

  4. FZU 1919 -- K-way Merging sort(记忆化搜索)

    题目链接 Problem Description As we all known, merge sort is an O(nlogn) comparison-based sorting algorit ...

  5. 经典排序算法 - 归并排序Merge sort

    经典排序算法 - 归并排序Merge sort 原理,把原始数组分成若干子数组,对每个子数组进行排序, 继续把子数组与子数组合并,合并后仍然有序,直到所有合并完,形成有序的数组 举例 无序数组[6 2 ...

  6. 连续线性空间排序 起泡排序(bubble sort),归并排序(merge sort)

    连续线性空间排序 起泡排序(bubble sort),归并排序(merge sort) 1,起泡排序(bubble sort),大致有三种算法 基本版,全扫描. 提前终止版,如果发现前区里没有发生交换 ...

  7. 排序算法二:归并排序(Merge sort)

    归并排序(Merge sort)用到了分治思想,即分-治-合三步,算法平均时间复杂度是O(nlgn). (一)算法实现 private void merge_sort(int[] array, int ...

  8. Simplest Python K-Way Merging Sort|最简单的Python k路归并排序

    想做这个好长时间了,因为有一篇Dreamworks的论文<Coherent Out-of-Core Point-Based Global Illumination>提到了这个,一直没时间做 ...

  9. 归并排序(merge sort)

    M erge sort is based on the divide-and-conquer paradigm. Its worst-case running time has a lower ord ...

随机推荐

  1. Codeforces 920G - List Of Integers

    920G - List Of Integers 思路:容斥+二分 代码: #include<bits/stdc++.h> using namespace std; #define ll l ...

  2. 有关C#中List排序的总结

    这里有一篇文章作者总结的就比较详细: https://blog.csdn.net/jimo_lonely/article/details/51711821 在这里只记录一点: 对list或者数组中的数 ...

  3. 放弃 Tightvnc, 选择 Tigervnc

    构建headless vnc server ,我终于放弃了Tightvnc 基于以下原因: 1) 已知的Qt5的键盘映射问题,导致virtualbox 的使用出现困难 https://unix.sta ...

  4. HTML第三章总结

    在这一章节中,主要讲了 HTML 中众多的 element,element 就像在建筑房屋时候的材料,它可以分为两种: Block Element Inline ElementBlock Elemen ...

  5. 雷林鹏分享:XML 验证器

    XML 验证器 使用我们的 XML 验证器来对您的 XML 文件进行语法检查. XML 错误会终止您的程序 XML 文档中的错误会终止您的 XML 应用程序. W3C 的 XML 规范声明:如果 XM ...

  6. ADO.NET Entity Framework学习笔记(3)ObjectContext

    ADO.NET Entity Framework学习笔记(3)ObjectContext对象[转]   说明 ObjectContext提供了管理数据的功能 Context操作数据 AddObject ...

  7. 进程状态TASK_UNINTERRUPTIBLE

    进程拥有以下几种状态:就绪/运行状态.等待状态(可以被中断打断).等待状态(不可以被中断打断).停止状态和僵死状态. TASK_RUNNING: 正在运行或处于就绪状态:就绪状态是指进程申请到了CPU ...

  8. 全站从http升级到https(WordPress博客)

    最近几年HTTPS取代HTTP已经成为大趋势,HTTP是超文本传输协议,信息是明文传输的,而HTTPS是安全超文本传输协议,需要证书和提供安全连接,换句话说,HTTPS是嵌套了SSL加密的HTTP连接 ...

  9. Confluence 6 教程:空间高手

    这个教程将会带你如何在 Confluence 中创建和自定义空间,同时也包括如何删除空间,如果需要的话.通过这个教程,你将成为使用空间的高手. 你需要具有创建空间(Create space)和创建个人 ...

  10. 【转】XP系统远程桌面连接2012R2提示:远程计算机需要网络级别身份验证,而您的计算机不支持该验证

    一.背景 因对方客户的服务器是内网的,需要操作更新服务器的数据库表信息,因此远程对方客户办公司的电脑远程服务器:但是在远程桌面连接出现问题. 二.错误问题 错误问题:“远程计算机需要网络级别身份验证, ...