数据结构与算法(二):排序算法

十大基础排序算法。

Basic-Sorting-Algorithm

关于十大基本排序算法的整理。

十大排序算法分别为:冒泡排序,选择排序,插入排序,希尔排序,堆排序,快速排序,归并排序,计数排序,桶排序和基数排序。

排序算法根据相同的值在排序之前和排序之后的前后位置是否不变来表示该排序算法是否稳定,如果不变则是稳定的,否则是不稳定的。

稳定:冒泡排序,插入排序,归并排序,计数排序,桶排序,基数排序

不稳定:选择排序,希尔排序,堆排序,快速排序

排序算法根据排序时所需数据是否一定要全部加载进内存来区分内外排,不需要则是外排,需要则是内排,注意,外排在数据少的时候也可以将需要排序的数据一次性全加载进内存,并不是外排就不可以处理数据少的情况,只是效率高与效率低的问题。

内排:冒泡排序,选择排序,插入排序,希尔排序,堆排序,快速排序

外排:归并排序,计数排序,桶排序,基数排序

Sorting-Algorithm.png

计数排序中的k指的是最大数值和最小数值的差值。

桶排序中的k指的是分成多少个桶。

基数排序中的k指的是进制中的基数,比如:十进制就是10。

基数排序中的d指的是最大数值的位数,比如:max=1000,则d=4。

希尔排序的时间复杂度比较有争议性,按照我的理解,希尔排序本质上也是插入排序的一种,也就是当增量=1时,希尔排序最好和最坏分别是:O(n)和O(n2)。

桶排序虽然排序需要遍历k遍,但是由于每个桶可以采取不同的排序方法,比如:统一采取平均时间复杂度为O(nlogn)的排序方法,则平均时间复杂度为:O(O(n+n(logn-logk)))。不仅时间复杂度和桶里数据采取的排序算法有关,连稳定性也是,比如,采取不稳定的算法,就有可能是不稳定的排序算法。

基数排序虽然将排好的数据重新写回去需要遍历k遍,但是,其实还是需要访问n个数据,其实时间复杂度可以写成:O(dn)。


1.冒泡排序

两两相邻的数据比较,如果前面的数据比后面的数据大,则交换两个数据的位置,直到所有的数据有序。

//Swift
func bubbleSort(sortedList: inout [Int]) {
    var i: Int = 1
    var flag = true  //优化

    while i < sortedList.count && flag {
        flag = false
        for j in 0..<sortedList.count-i {
            if sortedList[j] > sortedList[j+1] {
                flag = true
                sortedList.swapAt(j, j+1)
            }
        }
        i++
    }
}

最好时间复杂度:最好的情况就是需要排序的数据完全有序,也就是只需要比较n-1次,移动0次,就可以得到一个完全有序的序列,所以时间复杂度为:O(n)。

最坏时间复杂度:需要排序的数据逆序,那么第一个数据需要比较n-1次,第二个数据需要比较n-2次,那么,总的比较时间为:n-1+n-2+n-3+...+1=(n^2 - n)/2,也就是时间复杂度为:O(n^2)。

平均时间复杂度:(O(n) + O(n^2))/2 = O(n^2)。

空间复杂度:因为排序是在原数组上进行交换和移动的,也就是不需要额外的辅助空间,严谨来说交换数据时需要一个临时的空间,所以空间复杂度为:O(1)。

稳定性:因为是相邻的元素两两比较,不存在跳跃比较,移动的情况,所以是稳定的排序。

排序类型:因为每次比较需要用到整个数组,换句话说需要把排序的数据一次性加载到内存里进行排序,所以是内排类型。

2.选择排序

选择排序,每次都在无序的数据中选出最大的数据,并排在后面,直到所有的数据有序。

//Swift
func simpleSelectSort(sortedList: inout [Int]) {
    for j in 0..<sortedList.count-1 {
        for i in j+1..<sortedList.count {
            if sortedList[j] > sortedList[i] {
                sortedList.swapAt(j, i)
            }
        }
    }
}

时间复杂度:选择排序比较特殊,无论排序的数据是有序还是无序,时间复杂度都是一样的。因为就算整个数据有序,但是你不将所有的数据比较一次,是不可能知道这个数据就是最大或者最小的,虽然人眼是能看出来,但是机器看不出,所以,时间复杂度为:O(n^2)。

空间复杂度:和冒泡排序一样,最多使用一个数据空间,所以空间复杂度为:O(1)。

稳定性:因为需要在剩下的所有数据中寻找最大值,存在跳跃的情况,比如:5 4 5 3 2 => 4 3 5 2 5 很明显前面的5跑到后面来了,所以是不稳定的。

排序类型:同冒泡排序一样,需要一次性把排序的数据加载到内存,所以是内排。

3.插入排序

插入排序是不断的将数据插入前面有序的序列,形成新的有序序列。

//Swift
func insertSort(sortedList: inout [Int]) {
    for j in 1..<sortedList.count {
        if sortedList[j] < sortedList[j-1] {
            let temp = sortedList[j]
            var i: Int = j-1
            while i >= 0 && sortedList[i] > temp {
                sortedList[i+1] = sortedList[i]
                i--
            }
            sortedList[i+1] = temp
        }
    }
}

最好时间复杂度:如果排序的数据完全有序,则只需要比较n-1次,不需要移动数据,则最好的时间复杂度为:O(n)。

最坏时间复杂度:如果排序的数据逆序,从第二数据开始,第一次在比较是否进入循环时,比较了一次,然后在循环比较移动时有比较了一次,也就是两次,总的时间复杂度为:2+3+4+...+n=(n+2)(n-1)/2,时间复杂度为:O(n^2)。

平均时间复杂度:O(n^2)。

稳定性:因为插入排序是一个一个数插入,也是相邻两个数据两两比较,不存在跳跃比较和移动的情况,所以是稳定的。

排序类型:同冒泡排序一样,需要一次性把排序的数据加载到内存,所以是内排。

4.希尔排序

希尔排序是插入排序的升级版,通过设置increment(增量),把数组分成increment组,分别进行插入排序。然后,increment不断的减少,最终一定是increment=1,也就是整个数组进行插入排序,得出有序的序列。

//希尔排序
func shellSort(sortedList: inout [Int]) {
    let length = sortedList.count
    var increment = length

    repeat{
        increment = increment/3+1
        for i in 0..<length {
            if i >= increment && sortedList[i] < sortedList[i-increment] {
                let temp = sortedList[i]
                var j: Int = i-increment
                while j >= 0 && sortedList[j] > temp {
                    sortedList[j+increment] = sortedList[j]
                    j -= increment
                }
                sortedList[j+increment] = temp
            }
        }
    }while increment > 1
}

时间复杂度:因为希尔排序是通过不同的增量来进行分组,然后每组进行插入排序的,也就是说增量的取值直接影响到希尔排序的时间复杂度。但是,在我看来,希尔排序说到底属于插入排,那么一开始increment=1,也就是希尔排序的最好时间复杂度和最坏时间复杂度都是和插入排序相同的,也就是O(n)~O(n^2) ,平均时间复杂度比较复杂,由于不同的增量取值,导致时间不一样,有时间复杂度为O(n^1.3) 和O(n^1.5)的增量取值,大家可以了解一下。

空间复杂度:和插入排序一样,空间复杂度为O(1)。

稳定性:因为希尔排序是增量插入排序,存在跳跃比较和移动的情况,所以是不稳定的排序。

排序类型:内排。

5.堆排序

堆排序是选择排序的升级版,通过一次次的构建大顶堆,不断获取堆中最大的数据,直到堆中没有数据,也就是所有数据都有序了。

//堆排序
 func heapSort(sortedList: inout [Int]) {
    sortedList.insert(sortedList.count, at: 0)
    let length = sortedList[0]

    for i in stride(from: length/2, through: 1, by: -1) {
        headAdjust(sortedList: &sortedList, index: i, length: length)
    }

    for i in stride(from: length, to: 1, by: -1) {
        sortedList.swapAt(1, i)
        headAdjust(sortedList: &sortedList, index: 1, length: i-1)
    }

    sortedList.removeFirst()
 }

 func headAdjust(sortedList: inout [Int], index: Int, length: Int) {
    let temp = sortedList[index]
    var s = index;  //根, index从1开始算
    var j = index*2  //左子树
    while j <= length {
        if j < length && sortedList[j] < sortedList[j+1] {
            j++
        }
        if temp >= sortedList[j] {
            break
        }
        sortedList[s] = sortedList[j]
        s = j;
        j = s*2  //左子树
    }
    sortedList[s] = temp
 }

最好时间复杂度:开始就是大顶堆,第一次构建,只需要比较,不需要移动,所有的数据至少需要比较一次,时间复杂度为:O(n)。然后,从第二次开始,由于每次都是取叶结点的数据取代根结点,所以,每次都需要比较和移动logi(i为当前构建大顶堆的结点数),也就是时间复杂度为:log(n-1)+log(n-2)+...+log(1)=log((n-1)!)=(n-1)log(n-1)(nlogn=logn!这个等式证明请自行百度),也就是时间复杂度为:O(nlogn)。

最坏时间复杂度:开始就是小顶堆,第一次构建,每个数据都需要比较和移动,也是O(n)的复杂度。从第二次开始,其实和最好情况的大顶堆是一样的,都需要比较和移动那么多的次数,时间复杂度都是:O(nlogn)。

空间复杂度:因为没有额外的辅助空间,所以,时间复杂度为:O(1)。

稳定性:因为存在跳跃的移动,所以是不稳定的排序。

排序类型:内排。

6.快速排序

快速排序是冒泡排序的升级版,归根到底是比较排序的一种。通过关键数,将数组分成左右两个数组,左边都小于关键数,右边都大于关键数,然后左右两个数组继续分下去,直到所有数据都有序。

 //快速排序
 func fastSort(sortedList: inout [Int]) {
    sort(sortedList: &sortedList, start: 0, end: sortedList.count-1)
 }

 func sort(sortedList: inout [Int], start: Int, end: Int) {
    if start < end {
        let m = partion(sortedData: &sortedList, start: start, end: end)
        sort(sortedList: &sortedList, start: start, end: m-1)
        sort(sortedList: &sortedList, start: m+1, end: end)
    }
 }

 //pivot = sortedData[start]
 func partion(sortedData: inout [Int], start: Int, end: Int) -> Int {
    let pivot = sortedData[start]
    var left = start
    var right = end

    while left < right {
        while left < right && sortedData[right] >= pivot {
            right--
        }
        sortedData.swapAt(left, right)
        while left < right && sortedData[left] <= pivot {
            left++
        }
        sortedData.swapAt(left, right)
    }
    return left
 }

最好时间复杂度:快速排序不断的把数组分成两边,相当于一棵二叉树,由二叉树的知识可以知道,完全二叉树的深度最小,为depth =?logn?+1,也就是说,当数据比较均匀的分布在二叉树的左右两边,则时间复杂度最小。假设快速排序的时间复杂度为:T(n),第一次需要遍历整个数据,然后把数据分成均匀的两部分,则时间复杂度为:T(n)=2T(n/2)+n,同理,T(n/2)=2T(n/4)+n/2,T(n/4)=2T(n/8)+n/8,则T(n)=2T(n/2)+n=2(2T(n/4)+n/2)+n=4T(n/4)+2n=4(2T(n/8)+n/4)+2n=8T(n/8)+3n=...=nT(n/n)+nlogn=nT(1)+nlogn=nlogn。因为完全二叉树的深度为logn,所以递归调用了logn次,并且直到分到叶子结点,也就是T(1),T(1)=0,所以,T(n)=nlogn。因此,快速排序的时间复杂度为:O(nlogn)。

最坏时间复杂度:由二叉树的知识可以直到,斜树的深度最大,为depth=n,也就是当整个数组元素构造成一棵斜树,那么,该时间复杂度最高。由最好时间复杂度得出的公式,可以用在最坏时间复杂度的计算,也就是:T(n)=T(n-2)+n-1=T(n-3)+n-1+n-2=T(n-4)+n-1+n-2+n-3=...=n-1+n-2+n-3+...+1=((n-1)*n)/2,所以,最坏时间复杂度为:O(n2)。

空间复杂度:最好的情况,需要进行logn次递归,所以空间复杂度为:O(logn),最坏的情况,需要进行n-1次递归,所以空间复杂度为:O(n),因此,空间复杂度为:O(logn)~O(n)。

稳定性:存在数据元素跳跃的问题,是不稳定的排序。

排序类型:内排。

7.归并排序

先没两个数据元素归并成一个有序的整体,然后有序的整体再两两归并成一个更大的有序整体,直到归并所有的数据元素,形成一个有序的整体。

 //归并排序
 func mergeSort(sortedList: inout [Int]) {
    var result: [Int] = Array.init(repeating: 0, count: sortedList.count)
    sort(sortedList: &sortedList, result: &result, start: 0, end: sortedList.count-1)
 }

 func sort(sortedList: inout [Int], result: inout [Int], start: Int, end: Int) {
    var result2: [Int] = Array.init(repeating: 0, count: MAXSIZE)
    if start == end {
        result[start] = sortedList[start]
    } else {
        let m = (end+start)/2
        sort(sortedList: &sortedList, result: &result2, start: start, end: m)
        sort(sortedList: &sortedList, result: &result2, start: m+1, end: end)
        merge(left: &result2, right: &result, start: start, middle: m, end: end)
    }
 }

 func merge(left: inout [Int], right: inout [Int], start: Int, middle: Int, end: Int) {
    var i = start, j = middle+1, k = start
    while i <= middle && j <= end {
        if left[i] < left[j] {
            right[k] = left[i]
            i++
        } else {
            right[k] = left[j]
            j++
        }
        k++
    }

    if i <= middle {
        for l in 0...middle-i {
            right[k+l] = left[i+l]
        }
    }
    if j <= end {
        for l in 0...end-j {
            right[k+l] = left[j+l]
        }
    }
 }

时间复杂度:因为两两归并,其实就是一棵完全二叉树,所以,最好和最坏的时间复杂度都是一样的,二叉树的深度为:logn,并且需要比较n次,所以为:O(nlogn)。

空间复杂度:需要n个额外的辅助空间存结果,并且需要递归logn次,所以空间复杂度为:O(n+logn),但是如果不采用递归,则需要:O(n)个空间。

稳定性:因为两个有序的整体merge的时候并不涉及到数据的跳跃比较和移动,所以是稳定的。

排序类型:外排,因为不需要刚开始就把所有的数据加载进内存进行排序。

8.计数排序

适用于整数,分布均匀的数据。先找到整个数组最小和最大的整数,然后生(max-min+1)长度的数组,遍历整个数组,最小的放在第一位,最大的放在最后一位,其他数据的位置根据和最小数据的差值放置相应小标的位置,只需遍历一遍就可以把整个数组的数据变成有序。

//计数排序
 func countSort(sortedList: inout [Int]) {
    let min = sortedList.min()!
    let max = sortedList.max()!
    var result = Array.init(repeating: 0, count: max-min+1)
        //排序
   for num in sortedList {
        result[num-min] = result[num-min]+1;
    }
    //打印
    var result2: [Int] = []
    for (index, value) in result.enumerated() {
        for _ in 0..<value {
            result2.append(index+min)
        }
    }
    print(result2)
 }

时间复杂度:排序时间复杂度为O(n),结果遍历时间复杂度为O(k)(k为最大和最小的差值+1),所以,时间复杂度为:O(n+k)。

空间复杂度:因为需要一个长度为k的数组接收结果,所以,空间复杂度为:O(k)。

稳定性:因为数据是一个个放进去的,相同数字的前后顺序是不变的,所以,是稳定的排序算法。

排序类型:外排,如果直到数据都是在哪一个数据段的,并不需要把所有的数据加载进内存,一个一个数据或者一部分一部分数据加载就可以了。

9.桶排序

计数排序的升级版,计数排序可以看作分成max-min+1个桶的排序。桶排序在计数排序的基础上,将max-min+1的数据段再分成k个桶,每个桶就是一个数据段,所有的桶数据段不会重叠,并且所有桶的数据段连起来就是max-min+1,先将所有数据加入桶里,然后桶里的数据再采用其他排序使桶里的数据有序,然后将所有桶的数据连接起来就是整个有序序列。

//桶排序
 func bucketSort(sortedList: [Int]) -> [Int] {
    let max = sortedList.max()!
    let min = sortedList.min()!
    let bucketSize = 20
    let bucketCount = (max-min)/bucketSize+1
    var buckets = Array.init(repeating: [Int](), count: bucketCount)
    for num in sortedList {
        let i = (num-min)/bucketSize
        var bucket = buckets[i]
        bucket.append(num)
        buckets[i] = bucket //因为Swift是用时复制,所以需要把bucket重新赋值回去
    }
    var result = [Int]()
    for var bucket in buckets {
        insertSort(sortedList: &bucket) //桶里采用插入排序
        result.append(contentsOf: bucket)
    }
    return result
 }

最好时间复杂度:整个数据均匀分布在n个桶里,时间复杂度为:O(n)。

最坏时间复杂度:所有数据都在一个桶里,则时间复杂度为:O(n2)。

平均时间复杂度:遍历需要n遍,排序需要k遍,即时间复杂度为:O(n+km)(m和桶采取的排序算法有关)。假如,桶采取的排序算法平均的时间复杂度为O(nlogn),则O(n+k(n/k)log(n/k))=O(n+n(logn-logk))=O(n+m)(m=n(logn-logk))。

空间复杂度:需要额外k个桶作为辅助,并且排序结果也需要n个位置储存数据,所以为:O(n+k)。

稳定性:因为桶里的排序用到的是插入排序,所有是稳定的。

排序类型:外排。

10.基数排序

将数字按不同的数位比较,从低位到高位,位数不足的补零。也就是先按照各位排序,然后再按照十位排序,再按照百位排序...直到最高位。

 func radixSort(sortedList: [Int]) -> [Int] {
    let max = sortedList.max()!
    var result = Array.init(sortedList)
    var buckets = Array.init(repeating: [Int](), count: 10)
    let maxDigit = "\(max)".count
    for i in 0..<maxDigit {  //d
        let mod = (pow(10, i+1) as NSDecimalNumber).intValue
        for num in result { //n
            let j = num%mod/(mod/10)
            var bucket = buckets[j]
            bucket.append(num)
            buckets[j] = bucket
        }
        var index = 0
        for j in 0..<buckets.count {  //k
            let bucket = buckets[j]
            for k in 0..<bucket.count {
                result[index+k] = bucket[k]
            }
            index = index+bucket.count
            buckets[j] = []
        }
    }
    return result
 }

时间复杂度:由上面的代码上面可以知道,时间复杂度和maxDigit有关,并且遍历整个数组需要n遍,然后将所有桶的数组按顺序加入结果数组,所以,还需要加上遍历桶的数据的次数,则和k有关。所以,时间复杂度为:O(d(n+k)),k为基数,比如十进制k就是10。

空间复杂度:因为用到了结果数据和基数个桶,所以为:O(n+k)。

稳定性:稳定的,因为不会改变相同两个数据的前后关系。

排序类型:外排。

友情链接:

Basic-Sorting-Algorithm
JS-Sorting-Algorithm

?著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,992评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,212评论 3 388
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事?!?“怎么了?”我有些...
    开封第一讲书人阅读 159,535评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,197评论 1 287
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,310评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,383评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,409评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,191评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,621评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,910评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,084评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,763评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,403评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,083评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,318评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,946评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,967评论 2 351

推荐阅读更多精彩内容