美文网首页
七种排序算法

七种排序算法

作者: 让我们荡起双桨呀 | 来源:发表于2020-03-19 17:11 被阅读0次

当n较大时,则应采用时间复杂度为O(nlogn)的排序算法:快速排序、堆排序、归并排序。

常见的排序算法有:

  • 插入排序(直接插入、希尔排序)
  • 选择排序(简单选择、堆排序)
  • 交换排序(冒泡排序、快速排序)
  • 归并排序

当n较大时,则应选择时间复杂度为O(nlogn)的排序算法:快速排序、堆排序、归并排序
快速排序是目前基于比较的内部排序中被认为最好的方法,当待排的关键字是随机分布时,快速排序的平均时间最短
如果大体有序的话,可以选择插入排序。

插入排序:

public static void insertSort(int[] arr){
    //存放待插入元素
    int temp;
    for (int i = 1; i < arr.length; i++){
        temp = arr[i];
        int j;
        for (j = i - 1; j >= 0; j--){
            //如果符合条件就往后移位,为当前元素腾出空间
            if (arr[j] > temp){
                arr[j + 1] = arr[i];
            } else {
                break;
            }
        }
        arr[j + 1] = temp;
    }
}

希尔排序:
基于插入排序的思想,缩小增量排序(直接插入的改进版)

public static void shellSort(int[] arr){
    //获取数组长度,确定增量gap
    int len = arr.length;
    int gap = len / 2;
    int tmp;
    
    while(gap > 0){
        for(int i = gap; i < len; i++){
            tmp = arr[i];
            //获取前一个元素的索引
            int preIndex = i - gap;
            while(preIndex >= 0 && arr[preIndex] > tmp){
                arr[preIndex + gap] = arr[preIndex];
                preIndex -= gap;
            }
            arr[preIndex + gap] = tmp;
        }
        gap /= 2;
    }
}

选择排序:

public static void selectSrot(int[] arr){
    int tmp;
    for(int i = 0; i < arr.length; i++){
        for(int j = j + 1; j < arr.length; j++){
            if (arr[i] > arr[j]){
                swap;
            }
        }
    }
} 

堆排序:
借助完全二叉树进行排序,将数据构建成大根堆或者小根堆,然后将顶点的元素和最后的元素互换,然后继续构建大根堆的过程。
设一个节点为R[i],则左孩子节点为R[2 * i + 1],右孩子为R[2 * i + 2]
最后一个非叶子节点为R[(len - 1) / 2]

public static void heapSort(int[] arr){
    //循环建立初始堆
    for(int i = (arr.length - 1) / 2; i >= 0; i--){
        headAdjust(arr, i, arr.length);
    }
    
    //进行n - 1次循环,完成排序
    for(int i = arr.length - 1; i--){
        //最后一个元素和第一个元素交换
        int tmp = arr[0];
        arr[0] = arr[i];
        arr[i] = tmp;
        
        //然后将将除最后一个元素继续构建大根堆
        headAdjust(arr, 0, i);
    }
}

public static void headAdjust(int[] arr, int parent; int len){
    //保存当前父节点
    int tmp = arr[parent];
    int child = parent * 2 + 1;
    
    while(child < len){
        //判断是否有右孩子,并且右孩子的值大于左孩子的值,则选择右孩子
        if (child + 1 < len && arr[child] < arr[child + 1]){
            child++;
        }
        //如果父节点的值大于孩子节点的值,则直接退出
        if (tmp >= arr[child]){
            break;
        }
        
        //把孩子节点的值赋值给父节点
        arr[parent] = arr[child];
        arr[child] = tmp;
    }
}

冒泡排序:

public static void bubbleSort(int[] arr){
    int tmp;
    for(int i = 1; i < arr.length; i++){
        for(int j = 0; j <= arr.length - 1 - i; j++){
            if (arr[j] > arr[j + 1]){
                swap;
            }
        }
    }
}

快速排序:
选取第一个元素作为基准元素,两端各设两个哨兵,右边移动到小于基准元素时停止,左边开始移动,直至遇到大于基准元素时停止,然后两元素交换,重复次过程,直至两哨兵相遇,将基准元素和相遇的值进行交换,此时基准元素左边的元素均小于基准元素,基准元素右边的值均大于基准元素,然后将基准元素左右两端的元素重复此过程,直至完全有序。

public static void quickSort(int left, int right){
    int i, j, t, tmp;
    if(left > right){
        return;
    }
    
    //基准元素
    tmp = arr[left];
    //左哨兵
    i = left;
    //右哨兵
    j = right;
    
    //顺序很重要,要先从左边开始
    while(i != j){
        while(arr[j] > tmp && i > j){
            j--;
        }
        while(arr[i] < tmp && i > j){
            i++;
        }
        if(i < j){
            t = arr[i];
            arr[i] = arr[j];
            arr[j] = t;
        }
    }
    //最终将基准元素归位
    arr[left] = arr[i];
    arr[i] = tmp;
    quickSort(left, i - 1);
    quickSort(i + 1, right);
}

归并排序:

public static void mergeSort(int[] arr, int left, int right){
    int mid = (right + left) / 2;
    if (left < right){
        mergeSort(arr, left, mid);
        mergeSort(arr, mid + 1, right);
        merge(arr, left, mid, right);
    }
}

public static void merge(int[] arr, int left, int mid, int right){
    //创建一个临时数组用于存放排序的数据
    int[] tmp = new int[right - left + 1];
    int i = left;  //左指针
    int j = mid + 1;  //右指针
    int k = 0;
    
    //将小的元素添加到临时数组中
    while(i < mid && j < right){
        if (arr[i] < arr[j]){
            tmp[k++] = arr[i++];
        } else {
            tmp[k++] = arr[j++];
        }
    }
    //将左边或右边剩余的元素添加到临时数组中
    while(i <= mid){
        tmp[k++] = arr[i++];
    }
    while(j <= right){
        tmp[k++] = arr[j++];
    }
    
    //将临时数组添加到原数组中
    k = 0;
    while(left <= right){
        arr[left++] = tmp[k++];
    }
}

相关文章

网友评论

      本文标题:七种排序算法

      本文链接:https://www.haomeiwen.com/subject/jpulyhtx.html