快排的3种方式

//(前两种时间复杂度为o(n^2) , 最后一种为o(n*logn)

public static void swap(int[] arr , int i , int j){
    arr[i] =arr[i] ^arr[j];
    arr[j] =arr[i] ^arr[j];
    arr[i] =arr[i] ^arr[j];
}
//使数组中以arr[R]划分,返回循环后arr[R]的所在地
public static int partition(int[] arr , int L ,int R ){
    if(L >R ){
        return -1;
    }
    if(L == R ){
        return L;
    }
    int lessEqual = L-1;
    int index = L;
    while (index <R ){
        if(arr[index] <=arr[R]){
            swap(arr ,index++ , ++lessEqual);
        }
    }
    swap(arr , ++lessEqual , R);
    return lessEqual;
}

//把一个数组以arr[R]划分,返回的是值为arr[R]的区间
public static int[] netherlandsFlag(int[] arr , int L , int R){
    if(L>R){
        return new int[] { -1 ,-1};
    }
    if(L ==R){
        return new int[] {L ,R};
    }
    int less = L-1;
    int more =R;
    int index = L;
    while (index <more){
        if(arr[index] ==arr[R]){
            index++;
        }else if(arr[index] <arr[R]){
            swap(arr ,index++ , ++less);
        }else{
            swap(arr ,index , --more);
        }
    }
    swap(arr ,more ,R );
    return new int[] {less+1 , more};
}
//递归1
public static void process1(int[] arr ,int L ,int R ){
    if(L >=R){
        return;
    }
    int M =partition(arr ,L ,R);
    process1(arr , L , M-1);
    process1(arr , M+1 ,R);
}
//快排1
public static void quickSort1(int[] arr){
    if(arr ==null || arr.length <2){
        return;
    }
    process1(arr ,0 , arr.length-1);
}

//递归2
public static void process2(int[] arr ,int L ,int R){
    if(L >=R){
        return;
    }
    int[] equalArea =netherlandsFlag(arr ,L ,R);
    process2(arr ,L ,equalArea[0] -1);
    process2(arr , equalArea[1] , R);
}
//快排2
public static void quickSort2(int[] arr){
    if(arr ==null || arr.length <2){
        return;
    }
    process2(arr ,0 , arr.length-1);
}

//递归3
public static void process3(int[] arr , int L ,int R){
    if(L > R ){
        return;
    }
    swap(arr ,L + (int) (Math.random() * (R - L+1)), R);
    int[] equalArea = netherlandsFlag(arr , L ,R);
    process3(arr , L , equalArea[0] -1);
    process3(arr , equalArea[1] +1, R );
}
//快排3
public static void quickSort3(int[] arr){
    if(arr == null ||arr.length <2){
        return;
    }
    process3(arr , 0  , arr.length-1);
}
  • 11
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
快速排序是一经典的排序算法,其主要思想是通过分治法将一个数组分为两个子数组,然后递归地对两个子数组进行排序。 以下是快速排序的几常见实现方式: 1. Hoare算法 Hoare算法是最初实现快速排序的算法,它使用两个指针分别从数组的两端开始扫描,当左指针指向的元素大于基准元素时,停止扫描;当右指针指向的元素小于基准元素时,停止扫描,然后交换左右指针所指向的元素,最后将基准元素放到最终的位置上。 ``` void quicksort(int arr[], int left, int right) { if (left >= right) return; int i = left, j = right, pivot = arr[left]; while (i <= j) { while (arr[i] < pivot) i++; while (arr[j] > pivot) j--; if (i <= j) { swap(arr[i], arr[j]); i++; j--; } } quicksort(arr, left, j); quicksort(arr, i, right); } ``` 2. Lomuto算法 Lomuto算法是另一常见的快速排序实现方式,它使用单个指针从左向右扫描数组,并将小于基准元素的元素交换到数组的左侧,最后将基准元素放到最终的位置上。 ``` void quicksort(int arr[], int left, int right) { if (left >= right) return; int i = left, j = left; for (; j < right; j++) { if (arr[j] < arr[right]) { swap(arr[i], arr[j]); i++; } } swap(arr[i], arr[right]); quicksort(arr, left, i - 1); quicksort(arr, i + 1, right); } ``` 3. 双向扫描 双向扫描是一优化的快速排序算法,它使用两个指针分别从数组的两端开始扫描,并且每次交换的元素都是不同的。这算法可以减少交换次数,并且可以处理有大量重复元素的情况。 ``` void quicksort(int arr[], int left, int right) { if (left >= right) return; int i = left, j = right, pivot = arr[left + (right - left) / 2]; while (i <= j) { while (arr[i] < pivot) i++; while (arr[j] > pivot) j--; if (i <= j) { if (i < j) swap(arr[i], arr[j]); i++; j--; } } quicksort(arr, left, j); quicksort(arr, i, right); } ``` 无论使用哪实现方式,快速排序的平均时间复杂度为O(nlogn),最坏情况下的时间复杂度为O(n^2),空间复杂度为O(logn)。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值