关于排序算法的学习与总结1---冒泡、选择、插入、希尔排序

排序算法的介绍

排序算法(SortAlgorithm)也称排序,排序是将一组数据,依指定的顺序进行排列的过程

排序的分类:

1)内部排序: 指将需要处理的所有数据都加载到内部存储器(内存)中进行排序:

2)外部排序法: 数据量过大,无法全部加载到内存中,需要借助外部存储(文件等)进行排序

 排序算法的时间复杂度:

  1. 平均时间复杂度是指所有可能的输入实例均以等概率出现的情况下,该算法的运行时间。

  2. 最坏情况下的时间复杂度称最坏时间复杂度。一般讨论的时间复杂度均是最坏情况下的时间复杂度。这样做的原因是:最坏情况下的时间复杂度是算法在任何输入实例上运行时间的界限,这就保证了算法的运行时间不会比最坏情况更长。

  3. 平均时间复杂度和最坏时间复杂度是否一致,和算法有关。

冒泡排序

冒泡排序(Bubble Sorting)的基本思想是:通过对待排序序列从前向后(从下标较小的元素开始),依次比较相邻元素的值,若发现逆序则交换,使值较大的元素逐渐从前移向后部,就象水底下的气泡一样逐渐向上冒。

优化: 因为排序的过程中,各元素不断接近自己的位置,如果一趟比较下来没有进行过交换,就说明序列已经有序,因此要在排序过程中设置一个标志 flag 判断元素是否进行过交换。从而减少不必要的比较。

冒泡排序的过程:

  1. 一共进行数组的大小-1 次大的循环

  2. 每一趟排序的次数在逐渐的减少

  3. 如果我们发现在某趟排序中,没有发生一次交换,可以提前结束冒泡排序

代码实现:

public class BubbleSort {
    //冒泡排序一共进行数组长度减一次循环,每次循环里面又是一次循环
    //每一趟排序的次数都在逐渐减少(每一轮都是将最大的放在末尾),时间复杂度为0(n^2)
    //优化:当一趟排序中没有进行过一次交换,则说明已经是有序了,可以退出循环
    public static void main(String[] args) {
        int[] arr = {3, 9, -1, 10, 20};
        bubbleSort(arr);
    }

    private static void bubbleSort(int[] arr) {
        int temp;
        for (int i = 0; i < arr.length - 1; i++) {
            boolean flag = false; //标识变量,默认没有进行过交换
            for (int j = 0; j < arr.length - 1 - i; j++){
                if (arr[j + 1] < arr[j]){
                    flag = true;
                    temp = arr[j];
                    arr[j] = arr[j + 1];
                    arr[j + 1] = temp;
                }
            }
            if (!flag){
                break;
            }

            System.out.println("第"+ (i+1) +"趟排序后的结果为");
            System.out.println(Arrays.toString(arr));
        }
    }
}

选择排序

选择式排序属于内部排序法,是从欲排序的数据中,按指定的规则选出某一元素,再依规定交换位置后达到排序的目的。其基本思想是:第一次从 arr[0]~arr[n-1]中选取最小值,与 arr[0]交换,第二次从 arr[1]~arr[n-1]中选取最小值,与arr[1]交换,第三次从 arr[2]~arr[n-1]中选取最小值,与 am[2]交换,…,第i次从 arr[i-1]~arr[n-1]中选取最小值,与 arr[i-1]交换,…,第 n-1 次从arr[n-2]~am[n-1]中选取最小值与 arr[n-2]交换,总共通过 n-1 次,得到一个按排序码从小到大排列的有序序列。

选择排序的过程:

 

  1. 选择排序一共进行数组长度减一轮排序,每一轮排序中又是一个循环(每一轮都是将最小的放在操作数据的首位)

  2. 先假定当前这个数是最小数,然后与后面数进行比较,当发现有比当前数更小的数,则重新确定最小数,并且得到下标

  3. 当遍历到数字末尾时,就得到本轮最小数和下标,最后进行交换

代码实现: 

public class SelectSort {
    public static void main(String[] args) {
        int[] arr = {101, 24, 119, 1, -8, 18};
        System.out.println("排序前");
        System.out.println(Arrays.toString(arr));

        selectSort(arr);
        System.out.println("排序后");
        System.out.println(Arrays.toString(arr));
    }

    //选择排序一共进行数组长度减一轮排序,每一轮排序中又是一个循环(每一轮都是将最小的放在操作数据的首位)
    //先假定当前这个数是最小数,然后与后面数进行比较,当发现有比当前数更小的数,则重新确定最小数,并且得到下标
    //当遍历到数字末尾时,就得到本轮最小数和下标,最后进行交换
    public static void selectSort(int[] arr) {
        for (int i = 0; i < arr.length - 1; i++) {
            int min = arr[i];  //最小数
            int minIndex = i;  //最小数下标
            //1.遍历数组寻找最小数
            for (int j = i + 1; j < arr.length; j++) {
                if (min > arr[j]) {
                    min = arr[j];  //重新确定最小数,并且得到下标
                    minIndex = j;
                }
            }
            //2.交换
            if (minIndex != i) {
                arr[minIndex] = arr[i];
                arr[i] = min;
            }
            /*System.out.println("第"+ (i+1) +"趟排序后的结果为");
            System.out.println(Arrays.toString(arr));*/
        }
    }
}

 

插入排序

 插入式排序属于内部排序法,是对于欲排序的元素以插入的方式找寻该元素的适当位置,以达到排序的目的。其基本思想是:把n个待排序的元素看成为一个有序表和一个无序表,开始时有序表中只包含一个元素,无序表中包含有 n-1 个元素,排序过程中每次从无序表中取出第一个元素,把它的排序码依次与有序表元素的排序码进行比较,将它插入到有序表中的适当位置,使之成为新的有序表。

 

插入排序的过程:

 

 

代码实现: 

public class InsertSort {
    public static void main(String[] args) {
        int[] arr = {101, 24, 119, 1, -8, 18};
        insertSort(arr);
        System.out.println(Arrays.toString(arr));
    }

    public static void insertSort(int[] arr){
        int insertVal;
        int insetIndex;
        for (int i = 1; i < arr.length; i++) {
            insertVal = arr[i];  //待插入的值
            insetIndex = i - 1;  //待插入的值前一个元素的下标

            //找到要插入的位置,移动
            while (insetIndex >= 0 && insertVal < arr[insetIndex]){
                arr[insetIndex + 1] = arr[insetIndex];
                insetIndex--;
            }
            //完成插入
            arr[insetIndex + 1] = insertVal;
        }
    }
}

希尔排序

简单的插入排序可能会出现问题,比如:数组 arr= {2,3,4,5,6,1}这时需要插入的数 1(最小),当需要插入的数是较小的数时,后移的次数明显增多,对效率有影响。所以我们可以使用希尔排序进行改进,也是一种插入排序,也称为缩小增量排序其基本思想为:希尔排序是把元素按下标的一定增量分组,对每组使用直接插入排序算法排序;随着增量逐渐减少,每组包含的元素越来越多,当增量减至1时,整个文件恰被分成一组,算法便终止。

希尔排序的过程:

希尔排序又包括交换法移动法,其中移动法效率比较高,但是比较难理解

移动法发现逆序的情况不会马上交换,会将较大的元素后移,直到找到待插入的元素的位置

 

 代码实现:

 

public class ShellSort {
    public static void main(String[] args) {
        int[] arr = {8, 9, 1, 7, 2, 3, 5, 4, 6, 0};
        shellSort2(arr);
        System.out.println("排序后的数组为" + Arrays.toString(arr));
    }

    //1.交换法(效率低)
    public static void shellSort(int[] arr) {
        int temp;
        int count = 0;
        for (int gap = arr.length / 2; gap > 0; gap /= 2) {
            for (int i = gap; i < arr.length; i++) {
                for (int j = i - gap; j >= 0; j -= gap) {
                    // 如果当前元素大于加上步长后的那个元素,说明交换
                    if (arr[j] > arr[j + gap]) {
                        temp = arr[j];
                        arr[j] = arr[j + gap];
                        arr[j + gap] = temp;
                    }
                }
            }
            System.out.println("希尔排序第" + (++count) + "轮 =" + Arrays.toString(arr));
        }
        /*
        推导过程:
        // 希尔排序的第1轮排序
        // 因为第1轮排序,是将10个数据分成了 5组
        for (int i = 5; i < arr.length; i++) {
            // 遍历各组中所有的元素(共5组,每组有2个元素), 步长5
            for (int j = i - 5; j >= 0; j -= 5) {
                // 如果当前元素大于加上步长后的那个元素,说明交换
                if (arr[j] > arr[j + 5]) {
                    temp = arr[j];
                    arr[j] = arr[j + 5];
                    arr[j + 5] = temp;
                }
            }
        }

        System.out.println("希尔排序1轮后=" + Arrays.toString(arr));//


        // 希尔排序的第2轮排序
        // 因为第2轮排序,是将10个数据分成了 5/2 = 2组
        for (int i = 2; i < arr.length; i++) {
            // 遍历各组中所有的元素(共5组,每组有2个元素), 步长5
            for (int j = i - 2; j >= 0; j -= 2) {
                // 如果当前元素大于加上步长后的那个元素,说明交换
                if (arr[j] > arr[j + 2]) {
                    temp = arr[j];
                    arr[j] = arr[j + 2];
                    arr[j + 2] = temp;
                }
            }
        }

        System.out.println("希尔排序2轮后=" + Arrays.toString(arr));//

        // 希尔排序的第3轮排序
        // 因为第3轮排序,是将10个数据分成了 2/2 = 1组
        for (int i = 1; i < arr.length; i++) {
            // 遍历各组中所有的元素(共5组,每组有2个元素), 步长5
            for (int j = i - 1; j >= 0; j -= 1) {
                // 如果当前元素大于加上步长后的那个元素,说明交换
                if (arr[j] > arr[j + 1]) {
                    temp = arr[j];
                    arr[j] = arr[j + 1];
                    arr[j + 1] = temp;
                }
            }
        }

        System.out.println("希尔排序3轮后=" + Arrays.toString(arr));//
        */
    }

    //2.对交换式的希尔排序进行优化->移位法
    public static void shellSort2(int[] arr) {
        for (int gap = arr.length / 2; gap > 0; gap /= 2) {
            for (int i = gap; i < arr.length; i++) {
                int j = i;
                int temp = arr[j];  //待插入的值
                while (j - gap >= 0 && temp < arr[j - gap]) {
                    //移动
                    arr[j] = arr[j - gap];
                    j -= gap;
                }
                //while循环结束说明找到了temp要插入的位置
                arr[j] = temp;
            }
        }
    }
}

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值