Java 实现的各种经典的排序算法小Demo

由于有上机作业,所以就对数据结构中常用的各种排序算法都写了个Demo,有如下几个:

  • 直接插入排序
  • 折半插入排序
  • 希尔排序
  • 冒泡排序
  • 快速排序
  • 选择排序
  • 桶排序
    Demo下载地址


下面谈一谈我对这几个排序算法的理解:

插入类算法

对于直接插入排序:(按从小到大的顺序)
核心原理:
若数组中只有一个元素,那么这就已经是有序的了;若数组中元素个数为两个,我们只需要对他们进行比较一次,要么交换顺序,要么不交换顺序就可以实现数组的内容的有序化;但是当数组内的元素的个数为N个呢?又该如何?这就催化了这个直接插入排序算法,其核心就是利用了有序化数组的方式,认为再插入一个新的元素之前都是有序的,只需要从后往前进行查找(找到一个小于待插入数据的位置,记为position,然后把这个数据之后的元素全部向后迁移一个,再把待插入数据插入到position+1的位置即可。(小伙伴们可以想象一下为什么是position+1,因为position位置上的数据小于我们的待插入的数据啊,所以要插在Position的下一个位置上!)

public static void DirectoryInsert(int []array,int length){
        int p,i;
        for(p=1;p<length;p++){
            int temp=array[p];
            i=p-1;
            while(i>=0&&array[i]>temp){
                array[i+1]=array[i];
                i--;
            }
            array[i+1]=temp;
        }
    }


关于折半插入排序算法:
核心原理:
折半插入的核心原理仍然是基于有序表的插入算法,找到位置后,仍然采用插入的方式进行数据添加。但是较之于直接插入有很大的提升,那就是在查找插入位置上的优化,速度上稍微有了一定的提升,虽然在乱序的数据上有良好的效果,但是时间复杂度仍然很大O(n^2)。是稳定的算法。
下面是代码的实现:

private static void Half(int[] array, int length) {
        //p stands for the times of the sort
        int left,right,mid,p;
        for(p=1;p<length;p++){
            int temp=array[p];
            left=0;right=p-1;
            while(left<=right){
                mid=(left+right)/2;
                if(array[mid]>temp){
                    right=mid-1;
                }else{
                    left=mid+1;
                }
            }
            for(int i=p-1;i>=left;i--){
                array[i+1]=array[i];
            }
            array[left]=temp;
        }
    }


对于希尔排序:
核心原理:
希尔排序核心仍然是基于插入方式的,以逐步减小“步长”,采用“分治”的思想对每一个子序列进行排序。最终实现对整个序列的排序。
特点:希尔排序是不稳定的排序算法,会导致数据原始相对位置的改变。如果以步长为2计算,其时间复杂度可达到O(n^2),若数据足够长,步长也足够大那么时间复杂度将接近与O(n),但是一般认为其为O(n^1.3)。
代码实现:

private static void Shell(int[] array, int length) {
        // TODO Auto-generated method stub
        int d=length/2;
        while(d>=1){
            for(int k=0;k<d;k++){
                //to every sub,carry the direcly insert
                for(int i=k+d;i<length;i+=d){
                    int temp=array[i];
                    int j=i-d;
                    while(j>=k&&array[j]>temp){
                        array[j+d]=array[j];
                        j-=d;
                    }
                    array[j+d]=temp;
                }
            }
            d=d/2;
        }
    }

交换类排序算法



对于冒泡排序:
核心原理:
冒泡排序是我们接触比较早的一个排序算法,其原理就是对数据两两进行比较大小,并对符合要求的数据进行交换。循环n-1次,便可以对n 个数据实现排序。
特点:
时间复杂度O(n^2),由于数据发生交换时并没有发生原始位置的变化,所以冒泡排序算法是稳定的排序算法。
代码实现:

private static void Bubble(int[] array, int length) {
        // TODO Auto-generated method stub
        for(int i=0;i<length;i++){
            //expeclally the end case is "length-i"
            for(int j=1;j<length-i;j++){
                if(array[j-1]>array[j]){
                    int temp=array[j];
                    array[j]=array[j-1];
                    array[j-1]=temp;
                }
            }
        }
    }

对于冒泡排序,这里还有一个改进版的冒泡,是针对于特殊情况下的排序的处理,比如一个已经有序的序列如果再进行正常的冒泡的话,就会浪费时间,所以,如果一个序列已经是有序的,那么就应该跳出这个序列的冒泡,从而在一定程度上减少了时间的浪费。
代码实现:

private static void BubbleBetter(int[] array, int length) {
        // TODO Auto-generated method stub
        boolean flag=false;
        for(int i=0;i<length;i++){
            //expeclally the end case is "length-i"
            for(int j=1;j<length-i;j++){
                flag=false;
                if(array[j-1]>array[j]){
                    int temp=array[j];
                    array[j]=array[j-1];
                    array[j-1]=temp;
                    flag=true;
                }
            }
            if(flag){
                return;
            }
        }
    }


快速排序算法:
核心原理:
快速排序的原理是找到轴值pivot(这里有两种方式,从代码中可以清晰地看到,但最终结果都是一样的,那就是找到这个分割点,再递归的进行排序。
特征:
时间复杂度为O(nlogn,已2为底);
代码如下:

private static void Fast(int[] array, int left, int right) {
        // TODO Auto-generated method stub
        if (left < right) {
            int p = Partition1(array, left, right);
            Fast(array, left, p - 1);
            Fast(array, p + 1, right);
        }
//      if (left < right) {
//          int p = Partition2(array, left, right);
//          Fast(array, left, p - 1);
//          Fast(array, p + 1, right);
//      }
    }

    private static int Partition1(int[] array, int left, int right) {
        int pivot = array[left];
        while (left < right) {
            while (left < right && array[right] >= pivot) {
                right--;
            }
            array[left] = array[right];
            while (left < right && array[left] <= pivot) {
                left++;
            }
            array[right] = array[left];
        }
        array[left] = pivot;
        return left;
    }

    private static int Partition2(int[] array, int start, int end) {
        int pivot = array[start];
        int left = start, right = end;
        while (left <= right) {
            while (left <= right && array[left] <= pivot) {
                left++;
            }
            while (left <= right && array[right] >= pivot) {
                right--;
            }
            if (left < right) {
                Swap(array[right], array[left]);
                left++;
                right--;
            }
        }
        Swap(array[start], array[right]);
        return right;

    }

选择类排序算法



对于选择排序:
核心原理:
两轮循环,第一轮是选择的次数的记录,第二轮是目标查找。所谓目标查找,就是找到一个符合要求的值,记录其位置,然后在第一轮的循环中进行判断,将符合条件者进行交换,如此可实现排序的功能。
特征:
时间复杂度O(n^2),交换n-1次,比较了n^2次。是不稳定的排序算法。
代码:

private static void Select(int[] array, int length) {
        // TODO Auto-generated method stub
        for(int i=1;i<length;i++){
            int k=i-1;
            for(int j=i;j<length;j++){
                if(array[j]<array[k]){
                    k=j;
                }
            }
            if(k!=i-1){
                int temp=array[i-1];
                array[i-1]=array[k];
                array[k]=temp;
            }
        }
    }

归并类排序算法



对于归并排序:
核心原理:
若一个序列只有一个元素,则它是有序的,归并排序不执行任何操作。否则归并排序将执行下面的递归步骤:
1)先把序列划分为长度基本相等的子序列
2)对每个子序列归并并排序
3)把排好序的子序列合并为最终的结果。
特征:
时间复杂度O(nlogn),是不依赖于数据原始顺序的不稳定的排序算法。
代码如下:

private static void MergeFunction(int[] array,int start, int end) {
        // TODO Auto-generated method stub
        if(start<end){
            int mid=(start+end)/2;
            MergeFunction(array, start, mid);
            MergeFunction(array, mid+1, end);
            Merge(array,start,mid,end);
        }
    }

    private static void Merge(int[] array, int start, int mid, int end) {
        // TODO Auto-generated method stub
        int len1=mid-start+1,len2=end-mid;
        int i,j,k;
        //声明数组,分别保存子串信息
        int[] left=new int[len1];
        int[] right=new int[len2];
        for(i=0;i<len1;i++){//执行数据复制
            left[i]=array[i+start];
        }
        for(i=0;i<len2;i++){//执行数据复制
            right[i]=array[i+mid+1];
        }
        i=0;j=0;
        //执行合并操作
        for(k=start;k<end;k++){
            if(i==len1||j==len2){
                break;
            }
            if(left[i]<right[j]){
                array[k]=left[i++];
            }else{
                array[k]=right[j++];
            }
        }

        //若array[start,mid]还有待归并数据,则放到array后面
        while(i<len1){
            array[k++]=left[i++];
        }

        //对array[mid+1,end]见得数据执行同样的操作
        while(j<len2){
            array[k++]=left[j++];
        }
        //释放内存操作
        left=null;
        right=null;

    }


总结:
排序算法多种多样,在不同的情况下选择合适的排序算法能让你事半功倍。

时间: 2024-10-24 19:07:53

Java 实现的各种经典的排序算法小Demo的相关文章

java语言中哪一种排序算法用的最多?

问题描述 java语言中哪一种排序算法用的最多? java语言中哪一种排序算法用的最多?快速排序既然效率高,为什么我们还要用冒泡呢?冒泡的好处是什么? 解决方案 不能说快速排序一定效率高,对于有序的序列,归并排序的效率就更高.对于大量小整数的排序,基数排序不但效率高,而且占用内存少.各种排序有不同的使用场合.所以都要学习,而不是问哪种常用. 解决方案二: 冒泡排序是用来理解排序的思路的,快速排序是默认的java排序,但是稳定性极差,建议你去百度八大排序,从快速插入排序开始,系统的学习理解排序.

Java中常用的6种排序算法详细分解_java

排序算法很多地方都会用到,近期又重新看了一遍算法,并自己简单地实现了一遍,特此记录下来,为以后复习留点材料. 废话不多说,下面逐一看看经典的排序算法: 1. 选择排序 选择排序的基本思想是遍历数组的过程中,以 i 代表当前需要排序的序号,则需要在剩余的 [i-n-1] 中找出其中的最小值,然后将找到的最小值与 i 指向的值进行交换.因为每一趟确定元素的过程中都会有一个选择最大值的子流程,所以人们形象地称之为选择排序.举个实例来看看: 初始: [38, 17, 16, 16, 7, 31, 39,

Java实现的几个常用排序算法详细解读

排序算法很多地方都会用到,近期又重新看了一遍算法,并自己简单地实现了一遍,特此记录下来,为以后复习留点材料. 废话不多说,下面逐一看看经典的排序算法: 1.选择排序 选择排序的基本思想是遍历数组的过程中,以 i 代表当前需要排序的序号,则需要在剩余的 [i-n-1] 中找出其中的最小值,然后将找到的最小值与 i 指向的值进行交换.因为每一趟确定元素的过程中都会有一个选择最大值的子流程,所以人们形象地称之为选择排序. 举个实例来看看: 初始: [38, 17, 16, 16, 7, 31, 39,

用Java实现几种常见的排序算法

排序|算法 用Java语言实现的各种排序,包括插入排序.冒泡排序.选择排序.Shell排序.快速排序.归并排序.堆排序.SortUtil等. 插入排序: package org.rut.util.algorithm.support; import org.rut.util.algorithm.SortUtil;/** * @author treeroot * @since 2006-2-2 * @version 1.0 */public class InsertSort implements S

排序算法(Java)——那些年面试常见的排序算法

前言 排序就是将一组对象按照某种逻辑顺序重新排列的过程.比如信用卡账单中的交易是按照日期排序的--这种排序很可能使用了某种排序算法.在计算时代早期,大家普遍认为30%的计算周期都用在了排序上,今天这个比例可能降低了,大概是因为现在的排序算法更加高效.现在这个时代数据可以说是无处不在,而整理数据的第一步往往就是进行排序.所有的计算机系统都实现了各种排序算法以供系统和用户使用. 即使你只是使用标准库中的排序函数,学习排序算法仍然有很大的实际意义: - 排序算法往往是我们解决其他问题的第一步 - 排序

我是如何击败Java自带排序算法的

Java 8 对自带的排序算法进行了很好的优化.对于整形和其他的基本类型, Arrays.sort() 综合利用了双枢轴快速排序.归并排序和启发式插入排序.这个算法是很强大的,可以在很多情况下通用.针对大规模的数组还支持更多变种.我拿自己仓促写的排 序算法跟Java自带的算法进行了对比,看看能不能一较高下.这些实验包含了对特殊情况的处理. 首先,我编写了一个经典的快速排序算法.这个算法通过计算样本的平均值来估计整个数组的中心点,然后用作初始枢轴. 我借鉴了一些Java的思路来适当改进我的快速排序

C/C++中的经典排序算法总结

C/C++中的经典排序算法总结 在C/C++中,有一些经典的排序算法,例如:冒泡排序.鸡尾酒排序或双向冒泡排序(改进的冒泡排序).选择排序.直接插入排序.归并排序.快速排序.希尔排序和堆排序等等.下面对这些排序算法进行一一解析并给出示例代码以共享之. 1.冒泡排序 冒泡排序是最基本的排序算法,之所以称之为冒泡排序是因为在冒泡排序的过程中总是大数往前放,小数往后放,相当于气泡上升. 冒泡排序的基本原理是:依次比较相邻的两个数,将大数放在前面,小数放在后面. 影响冒泡排序算法性能的主要部分就是循环和

Java实现几种常见排序算法代码_java

稳定度(稳定性)一个排序算法是稳定的,就是当有两个相等记录的关键字R和S,且在原本的列表中R出现在S之前,在排序过的列表中R也将会是在S之前. 排序算法分类 常见的有插入(插入排序/希尔排序).交换(冒泡排序/快速排序).选择(选择排序).合并(归并排序)等. 一.插入排序 插入排序(Insertion Sort),它的工作原理是通过构建有序序列,对于未排序数据,在已排序序列中从后向前扫描,找到相应位置并插入.插入排序在实现上,通常采用in-place排序(即只需用到O(1)的额外空间的排序),

JavaScript中几种排序算法的简单实现_基础知识

排序算法的实现 我的JS水平就是渣渣,所以我就用类似于JAVA和C的方式来写JavaScript的排序算法了. 而且这里我不讲算法原理,仅仅只是代码实现,可能会有Bug,欢迎大家博客评论指导.插入排序 插入排序(Insertion-Sort)的算法描述是一种简单直观的排序算法.它的工作原理是通过构建有序序列,对于未排序数据,在已排序序列中从后向前扫描,找到相应位置并插入.插入排序在实现上,通常采用in-place排序(即只需用到O(1)的额外空间的排序),因而在从后向前扫描过程中,需要反复把已排