Go语言实现快排+随机快排

快排算法是一种本地算法,(即不需要额外的内存空间,就地排序)

  • 基本思想:
    从这个数列里找一个数作为基准点(支点)跟其它的数进行对比,小于或等于这个支点数的都放到左边,大的放到右边。完成一次排序,然后依次递归,当起始下标和末尾下标相遇是,排序结束,即整列数已经排列好了顺序。
  • 快排为什么快呢?
    简单的总结是:快排实现了最优下界。比如著名的称球问题12个小球有一个是坏的(轻或重), 现在呢有一架天平,问最少可用多少次就能确定哪个小球是坏的,并且指出它是轻了还是重了。答案是三次,称三次就能得出结论。等概率最优。时间复杂度为O(nlgn)(n乘以以10为底n的对数)
  • 快排为什么好像不那么快呢?
    当输入的数列是一个已经排序好的数列时(顺序或逆序),就慢了,因为它无法找到一个支点是两边等分的,无法做到等概率化,这是快排的最坏的情况,快排此时傻眼中。。。时间复杂度为O(n*n)但是没关系,下面第二种实现随机化快排算法可以弥补这个缺憾, 它不再关心你的输入数列是什么。

(1)基本快排算法,本示例用输入数列的第一个值作为排序支点。

package main

import (
    "fmt"
    "math/rand"
)

func main() {
    // get the random numbers
    origin := rand.Perm(10)
    fmt.Println("origin:", origin)
    quickSort(origin, 0, len(origin))
    fmt.Println("quick sort:", origin)

}

func quickSort(list []int, start, end int) {
    if end-start > 1 {
        // get the pivot
        mid := partition(list, start, end)
        // left sort
        quickSort(list, start, mid)
        // right sort
        quickSort(list, mid+1, end)
    }
}

func partition(list []int, begin, end int) (i int) {
    cValue := list[begin]
    i = begin
    for j := i + 1; j < end; j++ {
        if list[j] < cValue {
            i++ // 这里一定要先加1后在交换值,因为支点现在不必交换,现在i 和 j(小于支点和大于支点)正在划分边界
            list[j], list[i] = list[i], list[j]
            fmt.Println("list:", list)
        }
    }
/* 到此,i和j的边界已经划分完成,把i对应的值和支点对应的值交换后,就符合了快分的要求:i左边对应的值都小于等于且右边的都大于支点对应的值
此时i的值就是新的支点, 对应的值就是新的主元
*/
    list[i], list[begin] = list[begin], list[i]
    return i
}

结果:

root@slave2:~# go run quickSort.go 
origin: [9 4 2 6 8 0 3 1 7 5]
quick sort: [0 1 2 3 4 5 6 7 8 9]

(2)随机快速排序算法
这个需要让每次的主元(就是那个被比较值)随机。实现如下:

package main

import (
    "fmt"
    "math/rand"
    "time"
)

func main() {
    //origin := rand.Perm(20) //伪随机数
    origin := []int{1, 2, 3, 4, 5, 6, 7, 8, 9}
    fmt.Println("origin:", origin)
    randomQuickSort(origin, 0, len(origin))
    fmt.Println("quick sort:", origin)

}

func randomQuickSort(list []int, start, end int) {
    if end-start > 1 {
        // get the pivot
        mid := randomPartition(list, start, end)
        randomQuickSort(list, start, mid)
        randomQuickSort(list, mid+1, end)
    }
}

func randomPartition(list []int, begin, end int) int {
    // 生成真随机数
    i := randInt(begin, end)
    fmt.Println("random number:", i)
    // 下面这行是核心部分,随机选择主元, 如果没有此次交换,就是普通快排
    list[i], list[begin] = list[begin], list[i]
    return partition(list, begin, end)
}

func partition(list []int, begin, end int) (i int) {
    cValue := list[begin]
    i = begin
    for j := i + 1; j < end; j++ {
        if list[j] < cValue {
            i++
            list[j], list[i] = list[i], list[j]
        }
    }
    list[i], list[begin] = list[begin], list[i]
    return i
}

// 真随机数
func randInt(min, max int) int {
    rand.Seed(time.Now().UnixNano())
    return min + rand.Intn(max-min)
}
    

为了显示随机快排对有序数列的处理要优于普通快排,特意输入了0-10的有序数列,测试结果如下:
普通快排:

root@slave2:~# go run quickSort.go 
origin: [1 2 3 4 5 6 7 8 9]
random number: 6
random number: 8
random number: 2
random number: 7
random number: 5
random number: 7
random number: 6
random number: 7
quick sort: [1 2 3 4 5 6 7 8 9]

随机快排:

root@slave2:~# go run randomQuickSort.go 
origin: [1 2 3 4 5 6 7 8 9]
random number: 7
random number: 6
random number: 4
random number: 1
random number: 2
quick sort: [1 2 3 4 5 6 7 8 9]

参考链接:
快排及随机化算法
数学之美番外篇:快排为什么那样快

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 1 初级排序算法 排序算法关注的主要是重新排列数组元素,其中每个元素都有一个主键。排序算法是将所有元素主键按某种方...
    深度沉迷学习阅读 1,463评论 0 1
  • 排序算法基础 排序算法,是一种能将一串数据按照特定的排序方式进行排列的一种算法,一个排序算法的好坏,主要从时间复杂...
    jackyshan阅读 4,018评论 3 11
  • 版本记录 前言 将数据结构和算法比作计算机的基石毫不为过,追求程序的高效是每一个软件工程师的梦想。下面就是我对算法...
    刀客传奇阅读 5,215评论 4 72
  • 记得前段时间有人说过,“今天的努力工作,是为了当年吹过的牛逼”。其实这就是对坚持最好的验证,如果一个能够...
    功不唐捐2019阅读 259评论 0 3
  • 风轻烟尘聋 夜来雪寒灯 楼高人心远 苦志不言情
    剑雨黑石阅读 153评论 0 0