排序算法之选择、插入、冒泡、快速

三个类:

  

  

AbstractSortService :
package cn.zhi.sort;

public abstract class AbstractSortService {

    public void swap(int[] array, int i, int j) {
        if (i != j) {
            int temp = array[i];
            array[i] = array[j];
            array[j] = temp;
        }
    }

    public void print(int[] arr) {
        for (int i : arr) {
            System.out.print(" " + i);
        }
    }

    public void print(int[] arr, int middle) {
        print(arr);
        System.out.println(" __:" + middle);
    }

    public void print(int[] arr, int i, int middle) {
        print(arr);
        System.out.println(" __: " + i + ":" + middle);
    }

}
package cn.zhi.sort;

public class SortImpl extends AbstractSortService {

    /**
     * 选择排序,O(n^2),从你一个位置开始,依次选出未排序的最小值和游标交换,交换次数O(n)。
     *
     * @param arr
     */
    public void select(int[] arr) {
        for (int i = 0; i < arr.length; i++) {
            int minV = arr[i];
            int min = i;
            for (int j = i + 1; j < arr.length; j++) {
                if (arr[j] < minV) {
                    min = j;
                    minV = arr[j];
                }
            }
            swap(arr, i, min);
        }
    }

    /**
     * 插入排序,同样也是O(n^2) 一层一层地排好序,最好排到最外层结束。
     *
     * @param arr
     */
    public void insert(int[] arr) {
        for (int i = 0; i < arr.length; i++) {
            for (int j = i; j > 0 && arr[j] < arr[j - 1]; j--) {
                swap(arr, j, j - 1);
            }
            // for (int j = i; j > 0; j--) {
            // if (arr[j] < arr[j - 1]) {
            // swap(arr, j, j - 1);
            // } else {
            // break;// 只判断相邻两个元素
            // }
            // }
        }
    }

    /**
     * 冒泡排序, 交换相邻元素 O(n^2) 从第一颗气泡开始冒泡,内层循环结束:最大的气泡浮出水面
     *
     * @param arr
     */
    public void bubble(int[] arr) {
        for (int i = arr.length; i >= 0; i--) {
            for (int j = 1; j < i && arr[j] < arr[j - 1]; j++) {
                swap(arr, j, j - 1);
            }
        }
    }

    /**
     * 快速排序,复杂度 nlog2(n),算是对冒泡排序的一种优化
     *
     * @param arr
     * @param left
     * @param right
     */
    public void quick(int[] arr, int left, int right) {
        if (left >= right) {
            return;
        }
        int l = left, h = right;
        int m = l, middle = arr[m];

        while (l < h) {
            while (m < h && arr[h] > middle) {
                h--;
            }
            swap(arr, m, h);
            m = h;
            while (l < m && arr[l] <= middle) {//注意此处应该是<=,否则数组有相同元素时会死循环
                l++;
            }
            swap(arr, m, l);
            m = l;
        }
        quick(arr, left, m - 1); // 对低字段表进行递归排序
        quick(arr, m + 1, right); // 对高字段表进行递归排序
    }

}
package cn.zhi.sort;

public class Test {
    public static void main(String[] args) {
        int[] arr = new int[] { 41, 43,78,21,89, 3,233,81,0,42,93,89,12,};
        SortImpl sort = new SortImpl();

        sort.print(arr);
        System.out.println();

//        sort.select(arr);
//        sort.insert(arr);
//        sort.bubble(arr);
        sort.quick(arr, 0, arr.length-1);

        sort.print(arr);
    }
}

输出结果:

 41 43 78 21 89 3 233 81 0 42 93 89 12
 0 3 12 21 41 42 43 78 81 89 89 93 233
时间: 2024-12-20 13:14:29

排序算法之选择、插入、冒泡、快速的相关文章

九大排序算法及其实现- 插入.冒泡.选择.归并.快速.堆排序.计数.基数.桶排序

  闲着的时候看到一篇“九大排序算法在总结”,瞬间觉得之前数据结构其实都有学过,但当初大多数都只是老师随口带过,并没有仔细研究一下.遂觉:这是欠下的账,现在该还了.   排序按照空间分类: In-place sort不占用额外内存或占用常数的内存 插入排序.选择排序.冒泡排序.堆排序.快速排序. Out-place sort:归并排序.计数排序.基数排序.桶排序. 或者按照稳定性分类: stable sort:插入排序.冒泡排序.归并排序.计数排序.基数排序.桶排序. unstable sort

“《算法》第4版第2章‘排序’”:初级排序算法(选择、冒泡、插入、希尔)

<算法>第4版作者是Robert Sedgewick 和 Kevin Wayne. 1. 选择排序 选择排序可以说是最简单的排序方法.首先,找到数组中最小的那个元素:其次,将它与数组的第一个元素交换位置(如果第一个元素就是最小元素,那么它就和自己交换):再次,在剩下的元素中找到最小的元素,将它与数组的第二个元素交换位置.如此往复,直到将整个数组排序. 该书中提出一个命题:对于长度为N的数组,选择排序需要大约N2/2次比较和N次交换.  程序如下: 1 void SelectionSort::s

Java 常用的排序算法【选择、冒泡、快排】

选择排序: 简述:从数组的第一个元素开始,依次与其他所有的元素对比,如果比自身大或小(取决于升序或降序)交换位置. ChiceSort Code 冒泡排序: 简述:比较数组中两个相邻的元素,如果前者比较大则交换位置.像啤酒杯中的气泡一样,先漂上来最大的气泡,再漂上来第二大的气泡......... BubbleSort Code 快速排序: 简述:寻找一个基准(数组中的第一个或最后一个),表的两端同时向中间扫描,小在左,大在右.然后分别从基准两边进行递归排序. QuickSortCode 原文地址

八大内部排序算法(上)-冒泡、直接插入、简单选择、快速

八大内部排序算法(上)冒泡.直接插入.简单选择.快速 排序分为内部排序和外部排序,内部排序是数据记录在内存中进行排序,而外部排序是因排序的数据很大,一次不能容纳全部的排序记录,在排序过程中需要访问外存. 我们这里说说八大排序就是内部排序. 1.直接插入排序 将一个记录插入到已排序好的有序表中,从而得到一个新,记录数增1的有序表.即:先将序列的第1个记录看成是一个有序的子序列,然后从第2个记录逐个进行插入,直至整个序列有序为止. 要点:设立哨兵,作为临时存储和判断数组边界之用. 直接插入实现如下:

数据结构排序算法之选择排序

今天继续介绍一种排序算法:选择排序. 选择排序的基本思想就是从待排序列中选择出最小的,然后将被选出元素和序列的第一个元素互换位置(当前默认是升序排列),则互换完成后第一个元素就是整个序列的最小的元素,则一次选择排序结束.然后我们从剩下的子序列中选择出最小的,然后将该被选出来的元素和该子序列的第一个元素(即整个序列的第二个元素)互换位置,则当前整个序列的第二个元素就是当前序列中的次最小值,第二次选择排序结束.以此类推,直到该待排序列只剩下一个元素后,则整个序列有序. 具体过程如下图所示: 下面就不

排序算法之选择排序

一. 算法描述 选择排序:在一个长度为N的无序数组中,在第一趟遍历N个数据,找出其中最小的数值与第一个元素交换,第二趟遍历剩下的N-1个数据,找出其中最小的数值与第二个元素交换......第N-1趟遍历剩下的2个数据,找出其中最小的数值与第N-1个元素交换,至此选择排序完成. 二. 算法分析 平均时间复杂度:O(n2) 空间复杂度:O(1)  (用于交换和记录索引) 稳定性:不稳定 (比如序列[5, 5, 3]第一趟就将第一个[5]与[3]交换,导致第一个5挪动到第二个5后面) 三. 算法实现

STL中排序算法的选择

 当大多数程序员需要对一组对象进行排序的时候,首先想到的一个算法是sort.sort是一个非常不错的算法,但它也并非在任何场合下都是完美无缺的.有时候我们并不需要一个完全的排序操作.比如说,如果我们有一个存放Widget的矢量,而我们希望将质量最好的20个Widget送给最重要的顾客,按照顾客的重要程度送上不同质量的Widget,那么只需要排序出前20个最好的Widget,其他的Widget可以不用排序.在这种情况下,需要的是一种部分排序的功能,而有一个名为partial_sort的算法正好

【排序算法】选择排序(Selection sort)

0. 说明 选择排序(Selection sort)是一种简单直观的排序算法. 它的工作原理如下. 首先在未排序序列中找到最小(大)元素,存放到排序序列的起始位置,然后,再从剩余未排序元素中继续寻找最小(大)元素,然后放到已排序序列的末尾.以此类推,直到所有元素均排序完毕. 选择排序的主要优点与数据移动有关.如果某个元素位于正确的最终位置上,则它不会被移动.选择排序每次交换一对元素,它们当中至少有一个将被移到其最终位置上,因此对 n 个元素的表进行排序总共进行至多 n-1 次交换.在所有的完全依

初级排序算法之选择排序

初级排序算法 本质是对要排序的数组进行嵌套循环,内层循环负责局部的排序,外层循环负责剩余的无序元素的递减.所以你只要理解嵌套循环和比较大小就能很快的掌握初级排序算法. 选择排序 一个无序的数组 a = [0, 4, 6, 3, 8, 2, 3, 9], 你也可以把a的元素想象成任何现实中可比较的具体物体.例如,有10根长短不一的木条,我们如何对它们进行排序?一个最直接的思想,先拿出最短的放到最前面,在剩余的木条中再拿出最短的放在第二位...直到最后一根木条.从中我们可以看出,1. 我们需要再一次

【排序算法】选择排序

选择排序算法原理 选择排序算法时间复杂度分析 选择排序算法稳定性分析 选择排序算法C语言代码 #include <stdio.h> //交换两个元素的值 void swap(int* a, int* b) { int temp; temp = *a; *a = *b; *b = temp; } void selectionSort(int arr[], int length) { int i, j, maxIndex; for(i = length; i > 0; i--) { //假设