常用排序算法——归并排序法

归并排序采用的是分治思想,将数组不断分解为子数组,直到子数组只有一个元素,每次分解对应一个归并函数,归并函数可以将当前分解的两个子数组合并起来。有两种方式可以实现归并排序,第一种是递归方式实现的,代码如下:
#include <iostream>

static void merge(int *left, int llen, int *right, int rlen){
    int i = 0, j = 0, pos = 0;
    int total = llen+rlen;

    int *tmp = new int[total];

    while(true){
        if(i == llen){
            while(j != rlen){
                tmp[pos++] = right[j++];
            }
            break;
        }
        else if(j == rlen){
            while(i != llen){
                tmp[pos++] = left[i++];
            }
            break;
        }
        else{
            if(left[i] < right[j]){
                tmp[pos++] = left[i++];
            }
            else{
                tmp[pos++] = right[j++];
            }
        }
    }

    for(int count = 0; count != total; ++count){
        left[count] = tmp[count];
    }

    delete []tmp;
}

void merge_sort(int *array, int len){
    if(len > 1){
        int center = len/2;

        merge_sort(array, center);
        merge_sort(array+center, len-center);

        merge(array, center, array+center, len-center);
    }
}

int main(){
    int array[10] = {1, 3, 5, 7, 9, 0, 8, 6, 4, 2};

    merge_sort(array, 10);

    for(int i = 0; i != 10; ++i){
        std::cout << array[i] << " ";
    }
    std::cout << std::endl;

    return 0;
}


第二种是基于循环的非递归方式实现的,整体性能要高于递归方式实现的归并排序,代码如下:
#include <iostream>

void merge( int* array, int* tmp, size_t length, size_t step){
    size_t i, j, pos, i_stop, j_stop;
    size_t k = 0;
    size_t stop = length-step;

    while(k < stop){
        pos = k;
        i_stop = k+step-1;
        j_stop = (k+2*step)<(length)?(k+2*step-1):(length-1);

        for(i = k, j = k+step; i <= i_stop && j <= j_stop; ){
            if(array[i] < array[j]){
                tmp[pos++] = array[i++];
            }
            else{
                tmp[pos++] = array[j++];
            }
        }

        while(i <= i_stop){
            tmp[pos++] = array[i++];
        }

        while(j <= j_stop){
            tmp[pos++] = array[j++];
        }

        k += 2*step;
    }

    for(i = 0; i != length; ++i){
        array[i] = tmp[i];
    }
}

void merge_sort( int* array, size_t length ){
    int* tmp = new int[length ];
    size_t k = 1;
    while(k < length){
        merge(array, tmp, 10, k);
        k *= 2;
    }
    delete [] tmp;
}

int main(){
    int array[] = {9, 8, 7, 6, 5, 4, 3, 2, 1, 0};

    merge_sort(array, 10);

    for(int i = 0; i != 10; ++i){
        std::cout << array[i] << " ";
    }
    std::cout << std:: endl;
    return 0;
}

由于后一种方法使用的是循环而非递归,减少了函数调用及堆栈开销,效率上高于第一种,但编写起来比第一种实现方式麻烦。


本文链接:http://blog.csdn.net/girlkoo/article/details/17606331

本文作者:girlkoo

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值