排序算法超详细代码和知识点整理(java版)

排序

1、冒泡排序

​ 两层循环,相邻两个进行比较,大的推到后面去,一共比较“数组长度”轮,每一轮都是从第一个元素开始比较,每一轮比较都会将一个元素固定到数组最后的一个位置。【其实就是不停的把元素往后堆,数组剩余长度越来越少,直到堆到最后一个,数组都堆好排好序了】

if (arr == null || arr.length < 2) 
    return;
for (int e = arr.length - 1; e > 0; e--) {
    for (int i = 0; i < e; i++) {
        if (arr[i] > arr[i + 1]) {
            swap(arr, i, i + 1); } } }

2、选择排序

​ 也是比较**“数组长度-1”轮,但是每一轮中是将一个最值元素放在开头**,然后从这个更新过元素的位置往后继续开始此轮的选择最值过程【其实就是不停的从选择最小元素,然后往前堆,和冒泡正好相反,然后前面剩余空间越来越少,直到前面都堆好了排好序了

if (arr == null || arr.length < 2) 
    return;
for(int i=0; i<arr.length - 1; i++){
    int minIndex = i;
    for(int j=i+1; j<arr.length; j++){
        minIndex = arr[j] < arr[minIndex] ? j : minIndex;
    } 
    swap(arr,i,minIndex);
}

3、插入排序

​ 重点看当前索引指向的元素大小,去对比比当前索引小,也就是前面排好序的那一部分元素。找到合适的位置,然后插进去。这对比的次数明显变少,只用和【索引-1】前面的数进行比较。

if (arr == null || arr.length < 2) 
    return;
for(int i = 1; i<arr.length; i++)
    for(int j = i-1; arr[j+1]<arr[j] && j >= 0; j--)
        swap(arr,j,j+1);

4、快速排序

​ 本质是找个基准值,一般可以选取这段数组中的最后一个元素。然后将小于这个基准值的放在其左侧,大于的放在其右侧.

​ 在遍历数组时,一般采用三指针法来实现。具体来说,我们使用一个左指针指向数组的左边界,用一个右指针指向数组的右边界。另外一个指针指向元素,用来遍历。

  • 如果当前查找的索引值对应数据 arr [i] 小于基准值,那么这个值和左边界【左指针】的下一个进行交换,同时左指针右移;

  • 如果大于基准值,就将其和右边界的前一个元素交换,同时左边界指针不动。索引 i 也不动,但是右指针左移

  • 如果等于基准值,索引 i ++

在这里插入图片描述

public static void quickSort(int[] arr){
    if(arr == null || arr.length < 2)
        return;
    quickSort(arr,0,arr.length-1);
}

public static void quickSort(int[] arr, int l, int r){
    if(l < r){
        int[] p = partition(arr,l,r);  //作用就是将最后一个元素当作基准值,然后将大于基准值得放右边,小于的放左边
        					//同时,返回一个基准信息,p[0]就是基准值中最左的一个【考虑到可能标志值会重复】,p[1]是最右侧
        quickSort(arr, l, p[0] - 1);
        quickSort(arr, p[1] + 1, r);
    }
}

public static void partiton(int[] arr, int l, int r){ //l是左边界,
	int pivot = arr[r];
    int left = l -1;
    int right = r;
    while(l < right){            //arr[l],就用来记录逐步遍历数组中的元素
        if(arr[l] < pivot)
            swap(arr, ++left, l++); // 将小于等于pivot的元素交换到左边界的右侧
        else if(arr[l] > pivot){
            swap(arr, --right, l); // 将大小于等于pivot的元素交换到右边界的左侧
        }
        else
            l++;
    }
    swap(arr, more, r); //最后将基准值转移到右指针位置
	return new int[] { less + 1, more };
}

5、归并排序

​ 将数组按照中间的位置,划分成左右两部分,然后对这两部分进行扫描归并。准备两个指针,分别指向左部分的开头,以及有部分的开头。然后比较大小,按照比较结果,将结果放进一个临时开辟的数组中,最后将这个临时数组的元素再塞回去。

public static void mergeSort(int[] arr){
	if(arr == null || arr.length < 2)
        return;
    mergeSort(arr,0,arr.length-1);
}

public static void mergeSort(int[] arr, int l, int r){
	if(l < r){
        int mid = l + ((r-l)>>1;
        mergeSort(arr, l, mid+1);
        mergeSort(arr, mid+1, r);     
        merge(arr,l,mid,r);
    }
}

public static void merge(int[] arr, int l, int mid, int r) {
 	int[] help = new int[r - l + 1];
    int i = 0;
    int p1 = l;
    int p2 = mid + 1;
    while(p1 <= mid && p2 <= r){
        help[i++] = arr[p1] < arr[p2] ? arr[p1++] : arr[p2++];
    }
    while (p1 <= m) {
        help[i++] = arr[p1++];
    }
    while (p2 <= r) {
        help[i++] = arr[p2++];
    }
    for (i = 0; i < help.length; i++) {
        arr[l + i] = help[i];
    }
}

6、堆排序

主要涉及两步骤:

1、构建堆【heapInsert】,2、输出堆顶元素,然后调整堆结构,以便于下一轮继续输出堆顶元素【heapify】

构建堆的过程就是不停的往上,也就是和自己的父节点、祖父节点…一直比较下去,如果想要大根堆,那就是只要大于父节点,那就和父节点交换,我们对数组的每一个元素都去单独构建一边。

调整堆,就是不停的往下。将堆顶元素和数组最后元素交换,由于当前数组经过构建堆过程后,已经是堆结构了。所以我们的堆顶元素肯定就是一个最值,交换过后,再重新调整堆即可。就是找到堆顶的孩子节点中最大的值,进行比较交换,一直比较到树的最后一层

public static void heapSort(int[] arr) {   //这个是主函数,参数中的arr数组中存放的就是待排元素
    if (arr == null || arr.length < 2) {
        return;
    }
    for (int i = 0; i < arr.length; i++) { //这个函数是用于创建大根堆,为了之后输出堆顶元素就是最大值
        heapInsert(arr, i);
    }
    int size = arr.length;    			  
    swap(arr, 0, --size);	//这一步是为了将堆顶元素和最后一个元素互换位置,注意最后要自减 也就是说最后应该数组中元素是从小到大排序的(如果一开始是一个大根堆)
    while (size > 0) {
        heapify(arr, 0, size);
        swap(arr, 0, --size); //同上面道理
    }
}
public static void heapInsert(int[] arr, int index) {   //这个函数比较好理解,就是一个循环,只要在创建堆结构过程中,新加入堆中的子节点比父节点元素大,就交换,最后形成的是大根堆
    while (arr[index] > arr[(index - 1) / 2]) {
        swap(arr, index, (index - 1) /2);
        index = (index - 1)/2 ;           //这一步别忘记了,一直是往下走的,index这个表示父节点位置的标记变量也要向下走
    }
}
public static void heapify(int[] arr, int index, int size) {//这次函数中参数中要借助 堆的大小size(对应上面主函数中的每次输出一次堆顶元素后,大小减1)
    int left = index * 2 + 1;
    while (left < size) {
        int largest = left + 1 < size && arr[left + 1] > arr[left] ? left + 1 : left;
        //上面这一步是为了找到左右子节点,哪个最大,然后记录最大的那个下标
        largest = arr[largest] > arr[index] ? largest : index;
        if (largest == index) {
            break;
        }
        swap(arr, largest, index);
        index = largest;
        left = index * 2 + 1;
    }
}
public static void swap(int[] arr, int i, int j) {       //就是一个交换函数,C++中直接用swap函数即可
    int tmp = arr[i];  arr[i] = arr[j];  arr[j] = tmp; }

7、优先级队列

默认是小根堆

//小根堆。默认
PriorityQueue<Integer> heap = new PriorityQueue<>();
  • 在左神算法讲解时举了一个例子,就是;

    堆排序扩展题目:
    已知一个几乎有序的数组,几乎有序是指,如果把数组排好顺序的话,每个元素移动的距离可以不超过k,并且k相对于数组来说比较小。请选择一个合适的排序算法针对这个数据进行排序。
    

    这时候就需要借助 堆结构 一开始先将前k个数存放入此优先级队列,也就是堆结构,然后将堆顶元素返回,也就是用pop返回

    这k个元素构成一个堆结构,找到这个堆中的小根堆的堆顶元素,就是当前最小的

    因为每个元素移动的距离不超过k,所以可以理解为这前k个数中最小的一个属肯也是整体数据中最小的一个数字,把第k+1个数放进去,再次形成小根堆,那返回的堆顶元素就是第二小的元素

    然后再重新从数组中取下一个元素,添加到堆中,自动就排好序形成相应的大根堆或者是小根堆

  • 下面我用左神算法中的java 代码来展示,其中关键的地方都加了注释了:

public void sortedArrDistanceLessK(int[] arr, int k) {
    PriorityQueue<Integer> heap = new PriorityQueue<>();  //java中的优先级队列的定义方式
    int index = 0;
    for (; index < Math.min(arr.length, k); index++) {
        //万一k超出范围了,就选取数组长度当作边界值,min()的作用就是这个
        heap.add(arr[index]);
    }
    int i = 0;
    for (; index < arr.length; i++, index++) {
        heap.add(arr[index]);  //上面那个for循环实际上没有取到第k个元素,所以先取
        arr[i] = heap.poll();  //然后再弹出堆顶元素
    }
    while (!heap.isEmpty()) {//数组最后一个元素也都加进堆结构后,剩下的就是将堆中的所有元素放依次弹出了
        arr[i++] = heap.poll();
    }
}

8、复杂度

  • 冒泡:时间复杂度o(n^2) 空间复杂度O(1)(原地排序,不需要额外空间)
  • 选择:时间复杂度o(n^2) 空间复杂度O(1)(原地排序,不需要额外空间)
  • 插入:时间复杂度o(n^2) 空间复杂度O(1)(原地排序,不需要额外空间)
  • 快速:时间复杂度o(nlogN) 空间复杂度平均情况下是O(log n),最坏情况下是O(n)。
  • 归并:时间复杂度o(nlogN) 空间复杂度O(n)(需要额外的空间来合并子数组)
  • :时间复杂度o(nlogN) 空间复杂度O(1)(原地排序)
  • 26
    点赞
  • 10
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值