排序算法(1)-冒泡排序,选择排序,插入排序,希尔排序,快速排序

一些基本的排序算法:

package tenSortingMethods;
/**
 * @author zhangdi
 * @description 排序算法
 * @Date 20180531
 */
public class Sort {

    public static void main(String[] args) {
        int[] arr = {6,1,2,7,9,3,4,5,10,8};//{ 26, 88, 45, 57, 12, 31, 12, 2, 64, 32, 20, 99, 1 };
        int[] arr1 = { 2, 1, 3, 5, 4 };// {6,1,2,7,9,3,4,5,10,8};

        // System.out.println(arr.getClass());
        System.out.println(" arr before sorting: " + IntArrtoString(arr));
        // BubbleSort(arr);
        // BubbleSort2(arr);
        // BubbleSort1_better(arr);
        // SelectSort(arr);
        // InsertionSort(arr);
        // ShellSort(arr);
        // ShellSort2(arr);

        // int[] quickSort = new Sort().QuickSort(arr, 0, arr.length - 1);
        // System.out.println(" quickSort after sorting: " +
        // IntArrtoString(quickSort));


         int[] quickSort2 = new Sort().QuickSort2(arr, 0, 9);
        // System.out.println(" quickSort2 after sorting: " +
        // IntArrtoString(quickSort2));

         System.out.println(" arr after sorting: " + IntArrtoString(arr));

    }

    /**
     * @description 冒泡排序
     * @param arr
     * @description : 两两比较,大的往后放; 每一次比较完成之后,下一次就少比较一个元素
     *              第一次比较有0个元素不比较;第二次有一个元素不需要比较;第三次有两个元素不需要比较;
     *              共需要比较arr.length-1次
     */
    public static void BubbleSort(int[] arr) {

        int temp;// 临时变量
        if (arr == null || arr.length == 0)
            return;
        for (int i = 0; i < arr.length - 1; i++) { // 表示趟数,一共arr.length-1次。
            for (int j = arr.length - 1; j > i; j--) {

                if (arr[j] < arr[j - 1]) {
                    temp = arr[j];
                    arr[j] = arr[j - 1];
                    arr[j - 1] = temp;
                }
            }
        }
    }

    /**
     * @description 冒泡排序-2
     * @param arr
     * 
     */
    public static void BubbleSort2(int[] arr) {

        int temp;// 临时变量
        if (arr == null || arr.length == 0)
            return;
        for (int i = 0; i < arr.length - 1; i++) { // 表示趟数,一共arr.length-1次。
            for (int j = 0; j < arr.length - 1 - i; j++) {

                if (arr[j] > arr[j + 1]) {
                    temp = arr[j];
                    arr[j] = arr[j + 1];
                    arr[j + 1] = temp;
                }
            }
        }
    }

    /**
     * @description 冒泡排序-1-优化
     * @param arr
     *            * 针对问题:数据的顺序排好之后,冒泡算法仍然会继续进行下一轮的比较,直到arr.length-1次,后面的比较没有意义的。
     *            方案: 设置标志位flag,如果发生了交换flag设置为true;如果没有交换就设置为false。
     *            这样当一轮比较结束后如果flag仍为false,即:这一轮没有发生交换,说明数据的顺序已经排好,没有必要继续进行下去。
     * 
     */
    public static void BubbleSort1_better(int[] arr) {

        int temp;// 临时变量
        boolean flag;
        if (arr == null || arr.length == 0)
            return;
        for (int i = 0; i < arr.length - 1; i++) { // 表示趟数,一共arr.length-1次。
            flag = false;
            for (int j = arr.length - 1; j > i; j--) {
                if (arr[j] < arr[j - 1]) {
                    temp = arr[j];
                    arr[j] = arr[j - 1];
                    arr[j - 1] = temp;
                }
            }
            if (!flag)
                break;
        }
    }

    /**
     * @description 选择排序
     * @param arr
     *            在长度为N的无序数组中,第一次遍历n-1个数,找到最小的数值与第一个元素交换;
     *            第二次遍历n-2个数,找到最小的数值与第二个元素交换; 。。。
     *            第n-1次遍历,找到最小的数值与第n-1个元素交换,排序完成。
     */
    public static void SelectSort(int[] arr) {
        int len = arr.length;
        int temp;// 临时变量
        if (arr == null || len == 0)
            return;
        for (int i = 0; i < len - 1; i++) {
            int minIndex = i;// 假定此时i位置元素为最小数值
            for (int j = i + 1; j < arr.length; j++) {
                if (arr[j] < arr[minIndex]) {
                    // 如果i+1后的元素值小于i位置的元素,那么改变minIndex值.当内层循环走完,此时数组最小值得小标就确定了
                    minIndex = j;
                }
            }
            if (minIndex != i) {
                // 最小值下标改变,交换元素
                temp = arr[i];
                arr[i] = arr[minIndex];
                arr[minIndex] = temp;
            }
        }
    }

    /**
     * @description 插入排序
     * @param arr
     * @description : 在要排序的一组数中,假定前n-1个数已经排好序,现在将第n个数插到前面的有序数列中,使得这n个数也是排好顺序的。
     *              如此反复循环,直到全部排好顺序。 位于表中后面的元素依次与表中前面的元素比较,若比之小,则还需继续和更前面的元素比较,
     *              直至遇到一个比它大的元素或者比较到第一个元素(哨兵)了。
     */
    public static void InsertionSort(int[] arr) {
        int len = arr.length;
        int temp;// 临时变量
        if (arr == null || len == 0)
            return;
        for (int i = 0; i < len - 1; i++) {// 趟数n-1趟
            for (int j = i + 1; j > 0; j--) {
                if (arr[j] < arr[j - 1]) {
                    temp = arr[j];
                    arr[j] = arr[j - 1];
                    arr[j - 1] = temp;
                } else {
                    break;
                }
            }
        }
    }

    /**
     * @description 希尔排序 (最小增量排序)
     * @param arr
     * @description: 我们选择增量gap=length/2,缩小增量继续以gap =
     *               gap/2的方式,这种增量选择我们可以用一个序列来表示,{n /2,(n/2)/2...1},称为增量序列。
     *               希尔排序的增量序列的选择与证明是个数学难题, 我们选择的这个增量序列是比较常用的,也是希尔建议的增量,
     *               称为希尔增量,但其实这个增量序列不是最优的。 此处我们做示例使用希尔增量。
     *               比如原始数组8917235460,初始增量为gap =length/5;即数组被分为五组
     *               [8,3],[9,5],[1,4],[7,6],[2,0] ,对五组分别进行插入排序;第一次排序后:
     *               3514089472,缩小增量为gap =gap/2 =
     *               2;分为两组[3,5,1,4,0],[8,9,4,7,2],插入排序 -->[0214357698],
     *               再缩小增量gap = gap/ = 1,[0214357698] -->插入排序.
     * 
     * 
     */
    public static void ShellSort(int[] array) {
        int temp = 0;
        int incre = array.length;
        while (true) {
            incre = incre / 2;
            for (int k = 0; k < incre; k++) { // 根据增量分为若干子序列
                for (int i = k + incre; i < array.length; i += incre) {
                    for (int j = i; j > k; j -= incre) {
                        if (array[j] < array[j - incre]) {
                            temp = array[j - incre];
                            array[j - incre] = array[j];
                            array[j] = temp;
                        } else {
                            break;
                        }
                    }
                }
            }

            if (incre == 1) {
                break;
            }
        }
    }

    /**
     * @description 希尔排序 (最小增量排序)
     * @param arr
     *            在希尔排序的理解时,我们倾向于对于每一个分组,逐组进行处理,但在代码实现中,
     *            我们可以不用这么按部就班地处理完一组再调转回来处理下一组( 这样还得加个for循环去处理分组)比如[5,4,3,2,1,0]
     *            ,首次增量设gap=length/2=3,则为3组[5,2] [4,1]
     *            [3,0],实现时不用循环按组处理,我们可以从第gap个元素开始,逐个跨组处理。同时,在插入数据时,
     *            可以采用元素交换法寻找最终位置, 也可以采用数组元素移动法寻觅。
     * 
     */
    /**
     * 希尔排序 针对有序序列在插入时采用交换法 见图
     * 
     * @param arr
     */
    public static void ShellSort2(int[] arr) {
        System.out.println("ShellSort2 start.........");
        // 增量gap,并逐步缩小增量
        for (int gap = arr.length / 2; gap > 0; gap /= 2) {
            // 从第gap个元素,逐个对其所在组进行直接插入排序操作
            System.out.println("1 gap:" + gap + "  arr:" + IntArrtoString(arr));
            for (int i = gap; i < arr.length; i++) {

                int j = i;
                while (j - gap >= 0 && arr[j] < arr[j - gap]) {
                    // 插入排序采用交换法
                    swap(arr, j, j - gap);
                    j -= gap;
                }
                System.out.println("2 gap:" + gap + " i:" + i + " arr:" + IntArrtoString(arr));
            }
        }
        System.out.println("ShellSort2 end...........");
    }

    /**
     * 希尔排序 针对有序序列在插入时采用移动法。
     * 
     * @param arr
     */
    public static void ShellSort3(int[] arr) {
        // 增量gap,并逐步缩小增量
        for (int gap = arr.length / 2; gap > 0; gap /= 2) {
            // 从第gap个元素,逐个对其所在组进行直接插入排序操作
            for (int i = gap; i < arr.length; i++) {
                int j = i;
                int temp = arr[j];
                if (arr[j] < arr[j - gap]) {
                    while (j - gap >= 0 && temp < arr[j - gap]) {
                        // 移动法
                        arr[j] = arr[j - gap];
                        j -= gap;
                    }
                    arr[j] = temp;
                }
            }
        }
    }

    /**
     * 交换数组元素
     * 
     * @param arr
     * @param a
     * @param b
     */
    public static void swap(int[] arr, int a, int b) {
        arr[a] = arr[a] + arr[b];
        arr[b] = arr[a] - arr[b];
        arr[a] = arr[a] - arr[b];
    }

    /**
     * @param arr
     * @description 快速排序 基本思想:(分治) 先从数列中取出一个数作为key值;
     *              将比这个数小的数全部放在它的左边,大于或等于它的数全部放在它的右边; 对左右两个小数列重复第二步,直至各区间只有1个数。
     * @ses http://developer.51cto.com/art/201403/430986.htm
     *      https://blog.csdn.net/morewindows/article/details/6684558
     */
    // 快速排序
    public int[] QuickSort(int[] a, int l, int r) {

        if (l < r) {
            int i = l, j = r, v = a[l];// i:左边开始活动坐标,j:右边开始活动坐标,v:基数,用于对比的值,一般取第一个
            while (i < j) {// 只要左边游标和游标不重叠的话那么就继续遍历
                System.out.println("i:" + i + " j:" + j + " arr:" + IntArrtoString(a));
                // 每一轮遍历的时候只进行一轮,把右边的一个比基数小的数移动到左边,把左边比基数大的数移动到右边
                while (i < j && a[j] > v)
                    // 找出右边比左边小的坐标
                    j--;
                if (i < j)
                    a[i++] = a[j];// 进行移动,左边坐标移动一个数
                System.out.println("i:" + i + " j:" + j + " arr:" + IntArrtoString(a));
                while (i < j && a[i] < v)
                    // 找出左边比右边大的坐标
                    i++;
                if (i < j)
                    a[j--] = a[i];// 进行移动,右边坐标移动一个数
                System.out.println("i:" + i + " j:" + j + " arr:" + IntArrtoString(a));
            }
            a[i] = v;// 当游标重叠时填入基数
            this.QuickSort(a, l, i - 1);
            this.QuickSort(a, i + 1, r);
            return a;
        }
        return a;
    }

    /**
     * @param a
     * @param left
     * @param right
     * @return  QuickSort2在一轮对比基准数的过程中不移动基准数,减少移动次数
     */
    public int[] QuickSort2(int[] a, int left, int right) {
        //校验要初始化变量之前;不校验易报错:java.lang.ArrayIndexOutOfBoundsException
        if (left>right ) {
            return a;
        }
        int i = left, j = right, s = a[left];
        int temp;

        while (i < j) {
            //System.out.println("1--------i:" + i + " j:" + j + " arr:" + IntArrtoString(a));
            while (a[j] >= s && j > i) {
                j--;
                //System.out.println("2--------i:" + i + " j:" + j + " arr:" + IntArrtoString(a));
            }
            while (a[i] <= s && j > i) {
                i++;
                //System.out.println("3--------i:" + i + " j:" + j + " arr:" + IntArrtoString(a));
            }
            if (i < j) {
                temp = a[j];
                a[j] = a[i];
                a[i] = temp;
                //System.out.println("4---------i:" + i + " j:" + j + " arr:" + IntArrtoString(a));
            }
        }
        a[left] = a[i];
        a[i] = s;

        QuickSort2(a, left, i - 1);
        QuickSort2(a, i + 1, right);
        return a;

    }



}
posted @ 2018-06-06 10:12  XueXueLai  阅读(171)  评论(0编辑  收藏  举报