合并排序

简介:

分治算法:

用分治策略实现n个元素进行排序的方法。

基本思想:

将待排序元素分成大小大致相同的两个子集合,分别对两个子集合进行排序,最终排好序的子集合合并成所要求的排好序的集合。

源码:

复制代码
/*
 * mergeSort.cpp
 *  合并排序算法,算法导论P.17
 *  Created on: 2011-12-21
 *      Author: LiChanghai
 */
//#include <iostream>
#include <vector>
#include <iostream>
#include <iterator>
using namespace std;
#define FLT_MAX 1.0E38 //定义一个很大的值作为哨兵

//对于待排序的数组 A[p...r], 其子数组A[p...q],A[q+1...r]已排好序
//函数 subMerge(A, p, q, r), 将两个已排好序的子数组A[p...q],A[q+1...r]
//合并成一个有序的数组代替当前的数组A[p...r]
template<typename T>
void subMerge(vector<T> &array,
                         typename vector<T>::iterator iterBegin,
                         typename vector<T>::iterator iterBarrier,
                         typename vector<T>::iterator iterEnd)
{
    //创建两个数组,分别存放以iterBarrier为界线的array的左边部分和右边部分
    vector<T> arrayLeft(iterBegin, iterBarrier+1);
    vector<T> arrayRight(iterBarrier+1, iterEnd);

    //在两个数组尾部分别放一个“哨兵”
    T temp = T(FLT_MAX);
    arrayLeft.push_back(temp);
    arrayRight.push_back(temp);

    //定义分别指向两个数组的迭代器
    typename vector<T>::iterator iterLeft = arrayLeft.begin();
    typename vector<T>::iterator iterRight = arrayRight.begin();

    //定义指向原数组array的迭代器
    typename vector<T>::iterator iterArray = iterBegin;

    for(; iterArray != iterEnd; ++iterArray)
        if(*iterLeft < *iterRight) //如果左边小,将左边的值放入原数组
        {
            *iterArray = *iterLeft;
            ++iterLeft;
        }
        else //如果右边小,将右边的值放入原数组
        {
            *iterArray = *iterRight;
            ++iterRight;
        }
    return;
}

//函数 mergeSort(A, p, r) 调用subMerge对任意数组排序,p, r为下标
//mergeSort(A, p, r)首先将数组A分为两部分
//然后递归调用其本身对这两部分 分别排序
//依次递归下去,直到只剩2个数的时候完成这两个数的排序
//然后再层层返回调用处,将已排好序的子序列合并成更大的有序序列
//最后一次调用subMerge时完成数组的排序
template<typename T>
void mergeSort(vector<T> &vec,
                         typename vector<T>::iterator iterHead,
                         typename vector<T>::iterator iterTail)
{

    //测试 mergeSort 调用了多少次
    //static int times_mergeSort=0;
    //cout<<"the "<<++times_mergeSort<<" call mergeSort()"<<endl;

    //测试 subMerge 调用了多少次
    //static int times_subMerge=0;

    if(iterHead < iterTail-1)
    {
        //数组长度的一半(错误的方法)
        //typename vector<T>::size_type halfLength = (vec.size()-1)/2;

        //数组长度的一半(正确方法)
        typename vector<T>::difference_type halfLength=( (iterTail-iterHead)-1)/2;

        //定义一个迭代器指向数组 vec 中间位置
        typename vector<T>::iterator iterDivide = iterHead + halfLength;

        mergeSort(vec, iterHead, iterDivide+1); //递归调用自身对前半段排序
        mergeSort(vec, iterDivide+1, iterTail); //递归调用自身对后半段排序
        //cout<<"the "<<++times_subMerge<<" call subMerge()"<<endl;
        subMerge(vec, iterHead, iterDivide, iterTail); //将上面排好序的两段合并
    }
    return;

}
int main()
{
    
    vector<int> vec;
    vec.push_back(0);
    vec.push_back(9);
    vec.push_back(8);
    vec.push_back(7);
    vec.push_back(6);
    vec.push_back(5);
    vec.push_back(4);
    vec.push_back(3);
    vec.push_back(2);
    vec.push_back(1);
    int size = vec.size();
    int *phead = &vec[0];
    int *ptail = &vec[size];
    mergeSort(vec,phead,ptail);
    for(int i=0;i<10;i++)
        cout<<vec[i]<<" ";
    cout<<endl;
    return 0;
}
复制代码

运行结果:

本文转自博客园xingoo的博客,原文链接:合并排序,如需转载请自行联系原博主。
相关文章
|
3月前
排序之希尔,归并,快排
排序之希尔,归并,快排
19 0
算法:分治思想处理归并递归问题
算法:分治思想处理归并递归问题
|
8月前
|
存储 人工智能 搜索推荐
浅谈归并排序:合并 K 个升序链表的归并解法
在面试中遇到了这道题:如何实现多个升序链表的合并。这是 LeetCode 上的一道原题,题目具体如下:
87 0
浅谈归并排序:合并 K 个升序链表的归并解法
【算法】全排序I,全排序II-回溯算法中的树枝去重和树层去重理解
【算法】全排序I,全排序II-回溯算法中的树枝去重和树层去重理解
|
机器学习/深度学习 算法 API
算法排序5——归并排序&分治思想
算法排序5——归并排序&分治思想
125 0
算法排序5——归并排序&分治思想
|
算法
【排序】归并类排序—归并排序(逆序数问题)
在排序中,我们可能大部分更熟悉冒泡排序、快排之类。对归并排序可能比较陌生。然而事实上归并排序也是一种稳定的排序,时间复杂度为O(nlogn).
119 0
【排序】归并类排序—归并排序(逆序数问题)
|
算法 C语言 索引
|
算法 搜索推荐 Java
算法-合并排序
合并排序是基于分治算法原理的最流行的排序算法之一。使用合并排序算法,一个问题被分成多个子问题。每个子问题都是单独解决的。最后,将子问题组合起来形成最终解决方案。
204 0

热门文章

最新文章