温馨提示×

温馨提示×

您好,登录后才能下订单哦!

密码登录×
登录注册×
其他方式登录
点击 登录注册 即表示同意《亿速云用户服务条款》

一种递归实现的快速排序精讲

发布时间:2020-07-11 22:37:38 来源:网络 阅读:662 作者:小沄 栏目:开发技术

简介:

快速排序是个“综合素质”较好的排序,比如javaSE中的Arrays.sort()实现原理,也是用的是快速排序思想。下面就看看一种快速排序的递归实现方式


要点:

1,分治思想,把问题划分成可以与本问题处理方式相同的若干子问题,使用递归来解决。

   如排序问题,可以

    (1)把原数组A[p,q](原问题)划分成三个部分 :较小部分A[p,m-1]  中间元素x=A[m]  较大部分A[m+1,q] 

          这样把部分当做一个整体看待是有序的A[p,m-1] <  A[m]  <  A[m+1,q]

    (2)同理,无论是较小部分还是较大部分都可以继续按(1)操作继续划分得更细,直到每个部分的元素

          被划分得只有单个元素,原数组之间每个元素就有序了

特征:

1,快速排序是原址排序,子问题解决的时候,不需要整体再次合并排序结果

2,递归调用在非常的数据的时候可能会发生栈溢出(递归深度太大)

难点:

   如何划分成相对有序的三个部分?

      思路:

         (1)在原数组的某个好识别的位置取出一个数做中间元素x(部分2)。(一般取数组末元素,如x=A[q])

         (2)采取一个位置变量i来左划分界线,保证i上和i的左边都是小部分元素(部分1)

            (i的初始位置应当在问题范围的前一个位置,因为此时还没开始划分)

         (3)除了x元素,循环取出原数组中的每个元素A[j]与x做比较

             遇到不大于中间元素x要收集到界线上:

                 原界线i右移一个,此时i指向的元素是不确定性质的元素,

                 不确定元素和已经遇到的确定的小部分元素A[j]交换位置。(不确定换确定)

             遇到大于中间元素的,界线i不动,j继续向右扫描(i和j之间的定是大部分元素,部分3)

         (4)让中间元素交换到真正位置。一遍循环后,可以分出 小部分 和 大部分,

              此时只要在界线的右一个(即大部分中的某个元素)与中间元素交换即可让划分的三个部分满足排序关系。




图片描述:

一种递归实现的快速排序精讲


代码描述:


void quicklySort(int *arr , int start , int last){

    int mid;

    if(start<last){

        mid = partition(arr, start, last);//划分子问题

        quicklySort(arr, start , mid-1);//解决左边部分

        quicklySort(arr, mid+1, last);//解决右边部分

    }

}


int partition(int *arr, int start, int last){

    int x = arr[last];//取本段数组最后一个元素,稍后使其成为划分好的中间元素

    int i=0,j=0;

    i=start-1;//使用i来控制两个部分的分界,初始值在子问题范围的前一个

    for(j=start;j<= last-1; j++){

        if(arr[j]<= x){//如果j走过的值是属于较小部分

           ++i;//较小部分分界线多收集一个空间

           swap(arr[i],arr[j]);//把属于较小部分的值交换到这个空间(原址排序)

        }

        //如果是较大部分,让j继续向后滑行

    }

    swap(arr[i+1],arr[last]);//把选择参照的值交换到小部分和大部分之间

    return i+1;//返回划分好的中间位置

}




向AI问一下细节

免责声明:本站发布的内容(图片、视频和文字)以原创、转载和分享为主,文章观点不代表本网站立场,如果涉及侵权请联系站长邮箱:is@yisu.com进行举报,并提供相关证据,一经查实,将立刻删除涉嫌侵权内容。

AI