Java常用排序算法

日常操作中常见的排序方法有:冒泡排序、快速排序、选择排序、插入排序、希尔排序,甚至还有基数排序、鸡尾酒排序、桶排序、鸽巢排序、归并排序等。

冒泡排序

    /**
     * <strong>冒泡排序法</strong>
     *
     * 使用类似于冒泡的方法,将数组中的较小的元素冒泡到最后,从而实现对于数据从大到小的排列顺序
     *
     * 使用第一个元素,逐个同后面的元素比较,如果第一个元素小,则上浮,同比较的元素替换位置,将最大的元素替换到第一位
     * 然后继续从第二个元素开始,逐个同后面的元素比较,将第二大的元素排在第二位,直至排序到最后一位,就可以形成从大到小的排列顺序
     *
     * @param numbers 需要排序的数组
     */
    public static void bubbleSort(int[] numbers) {
        int temp, size = numbers.length;

        for (int i = 0; i < size - 1; i++) {
            for (int j = i + 1; j < size; j++) {
                if (numbers[i] < numbers[j]) {//如果前面的元素小,则上浮
                    temp = numbers[i];
                    numbers[i] = numbers[j];
                    numbers[j] = temp;
                }
            }
        }
    }

快速排序

    /**
     * <strong>快速排序法</strong>
     *
     * <ul>
     * <li>从数列中挑出一个元素,称为“基准”</li>
     * <li>重新排序数列,所有元素比基准值小的摆放在基准前面,所有元素比基准值大的摆在基准的后面(相同的数可以到任一边)。在这个分割之后,
     * 该基准是它的最后位置。这个称为分割(partition)操作。</li>
     * <li>递归地把小于基准值元素的子数列和大于基准值元素的子数列排序。</li>
     * </ul>
     *
     * @param numbers 需要排序的数组
     * @param start 开始位置
     * @param end 结束位置
     */
    public static void quickSort(int[] numbers, int start, int end) {
        if (start < end) {
            int base = numbers[start]; // 选定的基准值(第一个数值作为基准值)
            int temp; // 记录临时中间值
            int i = start, j = end;
            do {
                while ((numbers[i] < base) && (i < end))//在不超过结束位之前的数字小于基准数字的,不进行处理
                    i++;
                while ((numbers[j] > base) && (j > start))//在不小于开始位之前的数字大于基准数字的,不进行处理
                    j--;
                if (i <= j) {//一旦到达此处,则认为i位置的值大于等于基准值,j位置的值小于等于基准值,则替换i和j位置的值
                    temp = numbers[i];
                    numbers[i] = numbers[j];
                    numbers[j] = temp;
                    i++;
                    j--;
                }
            } while (i <= j);//在开始位置和结束位置之间
            if (start < j)//如果开始位置小于小于j判断的位置,则比较这两个区段内的数字,进行位置对换
                quickSort(numbers, start, j);
            if (end > i)//如果结束位置大于i判断的位置,则比较这两个区段的位置,进行位置对换
                quickSort(numbers, i, end);
        }
    }

选择排序

    /**
     * 选择排序
     * <ul>
     * <li>在未排序序列中找到最小元素,存放到排序序列的起始位置</li>
     * <li>再从剩余未排序元素中继续寻找最小元素,然后放到排序序列末尾。</li>
     * <li>以此类推,直到所有元素均排序完毕。</li>
     * </ul>
     *
     * @param numbers 需要排序的数组
     */
    public static void selectSort(int[] numbers) {
        int temp, size = numbers.length;
        for (int i = 0; i < size; i++) {//从第一个元素开始定位
            int k = i;
            for (int j = size - 1; j >i; j--)  {//从最后一个元素开始寻找
                if (numbers[j] < numbers[k])  k = j;//如果被寻找的元素,小于定位的元素,则替换二者的位置
            }
            temp = numbers[i];
            numbers[i] = numbers[k];
            numbers[k] = temp;
        }
    }

插入排序

    /**
     * 插入排序<br/>
     * <ul>
     * <li>从第一个元素开始,该元素可以认为已经被排序</li>
     * <li>取出下一个元素,在已经排序的元素序列中从后向前扫描</li>
     * <li>如果该元素(已排序)大于新元素,将该元素移到下一位置</li>
     * <li>重复步骤3,直到找到已排序的元素小于或者等于新元素的位置</li>
     * <li>将新元素插入到该位置中</li>
     * <li>重复步骤2</li>
     * </ul>
     *
     * @param numbers
     */
    public static void insertSort(int[] numbers) {
        int size = numbers.length, temp, j;
        for(int i=1; i<size; i++) {
            temp = numbers[i];
            for(j = i; j > 0 && temp < numbers[j-1]; j--)
                numbers[j] = numbers[j-1];
            numbers[j] = temp;
        }
    }

归并排序

    /**
     * 归并排序<br/>
     * <ul>
     * <li>申请空间,使其大小为两个已经排序序列之和,该空间用来存放合并后的序列</li>
     * <li>设定两个指针,最初位置分别为两个已经排序序列的起始位置</li>
     * <li>比较两个指针所指向的元素,选择相对小的元素放入到合并空间,并移动指针到下一位置</li>
     * <li>重复步骤3直到某一指针达到序列尾</li>
     * <li>将另一序列剩下的所有元素直接复制到合并序列尾</li>
     * </ul>
     *
     * @param numbers
     */
    public static void mergeSort(int[] numbers) {
        sort(numbers, 0, numbers.length - 1);
    }

    public static void sort(int[] numbers, int left, int right) {
        if (left >= right)
            return;
        // 找出中间索引
        int center = (left + right) / 2;
        // 对左边数组进行递归
        sort(numbers, left, center);
        // 对右边数组进行递归
        sort(numbers, center + 1, right);
        // 合并
        merge(numbers, left, center, right);
        System.out.println(Arrays.toString(numbers));
    }
    /**
     * 将两个数组进行归并,归并前面2个数组已有序,归并后依然有序
     *
     * @param numbers
     *            数组对象
     * @param left
     *            左数组的第一个元素的索引
     * @param center
     *            左数组的最后一个元素的索引,center+1是右数组第一个元素的索引
     * @param right
     *            右数组最后一个元素的索引
     */
    public static void merge(int[] numbers, int left, int center, int right) {
        // 临时数组
        int[] tmpArr = new int[numbers.length];
        // 右数组第一个元素索引
        int mid = center + 1;
        // third 记录临时数组的索引
        int third = left;
        // 缓存左数组第一个元素的索引
        int tmp = left;
        while (left <= center && mid <= right) {
            // 从两个数组中取出最小的放入临时数组
            if (numbers[left] <= numbers[mid]) {
                tmpArr[third++] = numbers[left++];
            } else {
                tmpArr[third++] = numbers[mid++];
            }
        }
        // 剩余部分依次放入临时数组(实际上两个while只会执行其中一个)
        while (mid <= right) {
            tmpArr[third++] = numbers[mid++];
        }
        while (left <= center) {
            tmpArr[third++] = numbers[left++];
        }
        // 将临时数组中的内容拷贝回原数组中
        // (原left-right范围的内容被复制回原数组)
        while (tmp <= right) {
            numbers[tmp] = tmpArr[tmp++];
        }
    }

归并算法因为需要创建新的数组,然后在排序完成后进行合并,因此会占用更多的内存空间,且随着数组内元素增长,数组拷贝和合并也会占用更多的时间。

参考:
Java实现几种常见排序方法

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 195,980评论 5 462
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 82,422评论 2 373
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 143,130评论 0 325
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,553评论 1 267
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,408评论 5 358
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,326评论 1 273
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,720评论 3 386
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,373评论 0 254
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,678评论 1 294
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,722评论 2 312
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,486评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,335评论 3 313
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,738评论 3 299
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,009评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,283评论 1 251
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,692评论 2 342
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,893评论 2 335

推荐阅读更多精彩内容

  • 分类:1)插入排序(直接插入排序、希尔排序)2)交换排序(冒泡排序、快速排序)3)选择排序(直接选择排序、堆排序)...
    小帝Ele阅读 1,145评论 1 18
  • 分类:1)插入排序 (直接插入排序、希尔排序)2)交换排序 (冒泡排序、快速排序)3)选择排序 (直接选择排序、堆...
    _凌浩雨阅读 249评论 0 1
  • 概述排序有内部排序和外部排序,内部排序是数据记录在内存中进行排序,而外部排序是因排序的数据很大,一次不能容纳全部的...
    Luc_阅读 2,250评论 0 35
  • 我的母亲很伟大。很多年来我一直想写点字句怀念母亲,但是我一是不擅长写,二是不知道写好后往何处安放。我把我的想...
    伊蒙伊阅读 736评论 6 3
  • 事情的起因是这样的 前两个月,我从校QQ群里加了个小学弟 人不错,也挺能干,马上大四毕业了,准备着开自己的工作室 ...
    大余说阅读 760评论 2 5