各种排序算法的Java实现


public class TestSort {
    //选择排序:它的工作原理是每一次从待排序的数据元素中选出最小(或最大)的一个元素,存放在序列的起始位置,直到全部待排序的数据元素排完。
    public static void selectSort(int[] a)
    {
        int minIndex=0;
        int temp=0;
        if((a==null)||(a.length==0))
            return;
        for(int i=0;i<a.length-1;i++)
        {
            minIndex=i;//无序区的最小数据数组下标
            for(int j=i+1;j<a.length;j++)
            {
                //在无序区中找到最小数据并保存其数组下标
                if(a[j]<a[minIndex])
                {
                    minIndex=j;
                }
            }
            if(minIndex!=i)
            {
                //如果不是无序区的最小值位置不是默认的第一个数据,则交换之。
                temp=a[i];
                a[i]=a[minIndex];
                a[minIndex]=temp;
            }
        }
    }
    
    //插入排序的基本思想是:每步将一个待排序的纪录,按其关键码值的大小插入前面已经排序的文件中适当位置上,直到全部插入完为止。
    //每次处理就是将无序数列的第一个元素与有序数列的元素从后往前逐个进行比较,找出插入位置,将该元素插入到有序数列的合适位置中。
    public static int[] insertSort(int[] arr){
        if(arr == null || arr.length < 2){
            return arr;//考虑空数组或者不需要排序的情况
        }
        for(int i=1;i<arr.length;i++){
            for(int j=i;j>0;j--){
                if(arr[j]<arr[j-1]){
                    //TODO:
                    int temp=arr[j];
                    arr[j]=arr[j-1];
                    arr[j-1]=temp;
                    }
                else{
                    //接下来是无用功
                    break;
                }
            }
        }
        return arr;
    }
    
    //归并排序:将已有序的子序列合并,得到完全有序的序列;即先使每个子序列有序,再使子序列段间/有序。
    //归并排序的算法我们通常用递归实现,先把待排序区间[s,t]以中点二分,接着把左边子区间排序,再把右边子区间排序,最后把左区间和右区间用一次归并操作合并成有序的区间[s,t]。
    public static void Merge(int array[], int p, int q, int r){
        int i, j, k, n1, n2;
        n1 = q-p+1;
        n2 = r-q;
        int[] L = new int[n1];
        int[] R = new int[n2];
        //左右两部分赋值
        for(i = 0,k=p;i<n1;i++,k++){
            L[i]=array[k];
        }
        for(i=0,k=q+1;i<n2;i++,k++){
            R[i]= array[k];
        }
        //取有序数组L/R中的较大者放入array直至其中一个全部取完
        for(k=p, i=0,j=0;i<n1 && j<n2; k++){
            if(L[i]>R[j]){
                array[k] = L[i];
                i++;
            }
            else{
                array[k] = R[j];
                j++;
            }
        }
        //将L/R剩下的全部复制arrayL中
        if(i<n1){
            for(j=i;j<n1;j++,k++){
                array[k] = L[j];
            }
        }
        if(j<n2){
            for(i=j;i<n2;i++,k++){
                array[k] = R[i];
            }
        }
    }
    
    public static void MergeSort(int array[], int p, int r){
        if(p<r){
            int q = (p+r)/2;
            MergeSort(array, p,q);
            MergeSort(array, q+1,r);
            Merge(array, p, q, r);
        }
    }
    
    //快速排序
    //通过一趟排序将要排序的数据分割成独立的两部分,其中一部分的所有数据都比另外一部分的所有数据都要小,
    //然后再按此方法对这两部分数据分别进行快速排序,整个排序过程可以递归进行,以此达到整个数据变成有序序列。
    
    public static void sort(int array[],int low, int high){
        int i, j;
        int index;
        if(low>=high){
            return;
        }
        i = low;
        j = high;
        index = array[i];                   //第一个元素作为枢轴
        while(i<j){                         //两端向中间扫描
            while(i<j && array[j]>=index){
                j--;
            }
            if(i<j){
                array[i++]=array[j];        //比枢轴小的交换到前段
            }
            while(i<j && array[i] <index){
                i++;
            }
            if(i<j){
                array[j--]=array[i];        //比枢轴大的交换到后端
            }
        }
        array[i]=index;
        sort(array,low,i-1);
        sort(array, i+1,high);
    }
    
    public static void quickSort(int array[]){
        sort(array, 0, array.length-1);
    }
    
    //希尔排序
    //先取一个小于n的整数d1作为第一个增量,把文件的全部记录分组。所有距离为d1的倍数的记录放在同一个组中。先在各组内进行直接插入排序;
    //然后,取第二个增量d2<d1重复上述的分组和排序,直至所取的增量   =1,即所有记录放在同一组中进行直接插入排序为止。
    
    public static void shellSort(int array[]){
        int length = array.length;
        int i, j;
        int h;
        int temp;
        for(h = length/2;h>0;h=h/2){            //增量变化
            for(i = h; i<length;i++){           //从i=h开始递增i
                temp = array[i];
                for(j = i-h;j>=0;j-=h){         //从i的位置,往前寻找前面的有序序列中的合适的位置 插入排序
                    if(temp<array[j]){
                        array[j+h]=array[j];                        
                    }
                    else{
                        break;
                    }
                }
                array[j+h]= temp;
            }
        }
    }
    
    /**
     * 希尔排序2
     * @param arrays 需要排序的序列
     */
    public static void sort(int[] arrays){
        if(arrays == null || arrays.length <= 1){
            return;
        }
        //增量
        int incrementNum = arrays.length/2;
        while(incrementNum >=1){
            for(int i=0;i<arrays.length;i++){
                //进行插入排序
                for(int j=i;j<arrays.length-incrementNum;j=j+incrementNum){
                    if(arrays[j]>arrays[j+incrementNum]){
                        int temple = arrays[j];
                        arrays[j] = arrays[j+incrementNum];
                        arrays[j+incrementNum] = temple;
                    }
                }
            }
            //设置新的增量
            incrementNum = incrementNum/2;
        }
    }
    
    
    /**堆排序
     * 堆排序利用了大根堆(或小根堆)堆顶记录的关键字最大(或最小)这一特征,使得在当前无序区中选取最大(或最小)关键字的记录变得简单。
     * 1. 建堆,建堆是不断调整堆的过程,从len/2处开始调整,一直到第一个节点,此处len是堆中元素的个数。
     * 2. 调整堆:调整堆在构建堆的过程中会用到,而且在堆排序过程中也会用到。
     * 3. 堆排序:交换栈顶元素和最后一个元素的位置
     * @param args
     */
    
    // a数组, pos调整起始位置, len调整终止位置
    public static void adjustMinHeap(int[] a, int pos, int len){
        int temp;
        int child;
        for(temp = a[pos]; 2*pos +1<=len; pos=child){
            child = 2 *pos +1;                              //左节点
            if(child <len && a[child]>a[child+1]){          
                child++;                                    //取小的节点
            }
            if(a[child]<temp){                              //顶点放最小的
                a[pos]= a[child];
            }
            else{
                break;                                      //不需要再继续递归
            }
        }
        a[pos] = temp;                                      //放回去
    }
    
    public static void myMinHeapSort(int[] array){
        int i;
        int len = array.length;
        //构建堆
        for(i = len/2-1;i>=0;i--){                  //注意这里要减去1才是正确的
            adjustMinHeap(array, i, len-1);
        }
        for(i=len-1;i>=0;i--){
            //交换
            int tmp = array[0];
            array[0] = array[len-1];
            array[i] = tmp;
            //调整堆
            adjustMinHeap(array, 0, i-1);
        }
        
    }
    
    
    public static void main(String[] args){
        int i = 0;
        int a[] ={5,4,9,8,7,6,0,1,3,2};
//      selectSort(a);
//      insertSort(a);
//      mergeSort(a, 0,1);
//      MergeSort(a, 0, a.length);
//      quickSort(a);
//      shellSort(a);
        sort(a);
        for(i=0;i<a.length;i++){
            System.out.println(a[i]+" ");
        }
        System.out.println("\n");
    }
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,039评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,223评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,916评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,009评论 1 291
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,030评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,011评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,934评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,754评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,202评论 1 309
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,433评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,590评论 1 346
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,321评论 5 342
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,917评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,568评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,738评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,583评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,482评论 2 352

推荐阅读更多精彩内容