归并排序学习

归并排序两种实现

归并排序的算法思路非常简单就是将待排序的数组不断分割,再对排好序的数组进行合并的过程。由此会有两个步骤:合并和分割,分割的过程主要是使用二分的思想不断进行分割,没有什么特殊的思路,合并过程则是遍历两个有序数组将其合并为一个数组,对此有两种不同的思路。
首先是分割过程两者是相同的:

    public void mergeSort(int[] arrays, int begin, int end) {
        if (begin < end) {
            int temp = (begin + end) / 2;
            mergeSort(arrays, begin, temp);
            mergeSort(arrays, temp + 1, end);
            merge(arrays, begin, temp, end);
        }
    }

一、普通的归并排序

    public void merge(int[] arrays, int begin, int temp, int end) {
        int n1 = temp - begin + 1;
        int n2 = end - temp;
        // 左边序列数组赋值
        int[] left = new int[n1];
        for (int i = 0; i < n1; i++) {
            left[i] = arrays[begin + i];
        }
        // 右边序列数组赋值
        int[] right = new int[n2];
        for (int i = 0; i < n2; i++) {
            right[i] = arrays[temp + 1 + i];
        }

        int i = 0, j = 0, k = 0;
        while (i <= left.length - 1 && j <= right.length - 1) {
            if (left[i] < right[j]) {
                arrays[begin + k++] = left[i++];
            } else {
                arrays[begin + k++] = right[j++];
            }
        }

        // 当有一个数组遍历完之后进行处理
        while (i <= left.length - 1) {
            arrays[begin + k++] = left[i++];
        }

        while (j <= right.length - 1) {
            arrays[begin + k++] = right[j++];
        }
    }

二、体现哨兵思想的归并排序

    public void merge(int[] arrays, int begin, int temp, int end) {
        int n1 = temp - begin + 2;
        int n2 = end - temp + 1;
        // 左边序列数组赋值
        int[] left = new int[n1];
        for (int i = 0; i < n1 - 1; i++) {
            left[i] = arrays[begin + i];
        }
        // 右边序列数组赋值
        int[] right = new int[n2];
        for (int i = 0; i < n2 - 1; i++) {
            right[i] = arrays[temp + 1 + i];
        }

        // 将左右数组大小扩大一个单元,用来存储哨兵
        left[n1 - 1] = right[n2 - 1] = Integer.MAX_VALUE;
        int m = 0;
        int n = 0;
        // 当一个数组遍历到哨兵,不可能有比哨兵更大的元素,当两个数组都遍历完之后,for循环结束
        for (int j = begin; j <= end; j++) {

            if (left[m] < right[n]) {
                arrays[j] = left[m];
                m++;
            } else {
                arrays[j] = right[n];
                n++;
            }
        }
    }
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值