排序算法大集合上

1. 插入排序

  1. 时间复杂度:最好是n-1,最坏是n(n-1)/2,平均为O(n^2)。
  2. 空间复杂度: O(1)
  3. 多作为快排的补充,适用于少量的数据排序。
  4. 该算法是稳定的,依赖初始排序顺序。

过程:

  1. 以第一个数为已经排好的队列,将第二个数从队列的右向左比较。
  2. 如果比它大,那么,该队列里的元素就往右边移动一位,接着比较该元素左边的元素;比它小,那么,就把他存入该队列里的元素的右边一位。
  3. 形成一个新的队列,再走1,2。
    更多
# 插入排序
def insert(nums):
    for i in range(1, len(nums)):
        temp = nums[i]
        # 是否找到一个合适的位置插入
        label = False
        for j in range(i-1,-1,-1):
            # 找到位置了,插入
            if nums[j] < temp:
                nums[j+1] = temp
                label = True
                break
            else:
                # 右移一位
                nums[j+1] = nums[j]
                
        if not label:
            nums[0] = temp

nums = [2,3,4,5,1,9,3,0,2,1]
insert(nums)
print(nums)

感觉上面这个方法有些复杂了,改进了一下:

def insert(nums):
    for i in range(1, len(nums)):
        j = i
        while j > 0:
            if nums[j] < nums[j - 1]:
                nums[j], nums[j - 1] = nums[j - 1], nums[j]
                j -= 1
            else:
                break

2. 二分插入排序

  1. 时间复杂度:最好情况下:O(nlogn);最坏情况下:O(n^2)
    分析:最外层有n次循环;最内侧分为两个部分,一个是二叉搜索(logn),一个是for循环(n);所以,最好的情况是元素所在的位置就是插入位置(nlogn);最坏情况和平均的情况就是不断在查找,即为:n(logn+n)->n^2。
  2. 空间复杂度维O(1)。
  3. 是稳定的, 依赖初始排序顺序。

过程:

  1. 在插入第i个元素时,对前面的0~i-1元素进行折半,
  2. 先跟他们中间的那个元素比,如果小,则对前半再进行折半。
  3. 否则对后半进行折半。回归2
  4. 直到left(左边缘)>right(右边缘),再把第i个元素前1位与目标位置之间的所有元素后移。
  5. 最后,把第i个元素放在目标位置上。
def insert_2_sort(nums):
    for i in range(1, len(nums)):
        left = 0
        right = i - 1
        key = nums[i]

        while left <= right:
            middle = (left+right)//2
            if key > nums[middle]:
                left = middle + 1
            else:
                right = middle - 1

        for j in range(i, left, -1):
            nums[j] = nums[j-1]

        nums[left] = key

a = [2,3,4,5,1,9,7,6,2,0,6]
insert_2_sort(a)

3. 希尔排序

  1. 空间复杂度为O(1)
  2. 时间复杂度则由增量(步长)而定。
    只要最终步长为1任何步长序列都可以工作。
    一个好的步长序列评价时间复杂度可以达到O(n^1.5)
  3. shell排序是非稳定排序
  4. 提高效率的思想:
    因为插入排序的时间复杂度依赖初始序列的顺序,所以通过大步长的排序(时间复杂度低),使序列部分有序,这样当进行小步长排序时,时间复杂度也低。

过程

  1. 把记录按下标的一定增量分组
  2. 对每组使用直接插入排序算法排序。
  3. 减小增量n,若n > 0,进入1;反之,算法终止
def shell(nums):
    gaps = 3
    for gap in range(1, gaps+1):
        for i in range(gap, len(nums)):
            key = nums[i]
            pointer = i - gap
            while key < nums[pointer] and pointer >= 0:
                nums[pointer+gap] = nums[pointer]
                pointer -= gap

            nums[pointer+gap] = key

4. 选择排序

  1. 空间复杂度为O(1)
  2. 时间复杂度为O(n^2)
  3. 对比冒泡排序,它少了很多的交换步骤。
  4. 不稳定的排序方式

过程:

  1. 从待排序的数据元素中选出最小(或最大)的一个元素。
  2. 存放在序列的起始位置,直到全部待排序的数据元素排完。
    类似冒泡排序。
def selection_sort(nums):
    for i in range(0, len(nums)):
        lowest = i
        for j in range(i, len(nums)):
            if nums[j] < nums[lowest]:
                lowest = j
        nums[lowest], nums[i] = nums[i], nums[lowest]

5. 冒泡排序

  1. 时间复杂度为O(n^2)
  2. 空间复杂度为O(1)
  3. 稳定的排序方法
  4. 优化:
    如果一个forloop没有发生一次交换,说明已经排好了,可以停止了后面的循环了。
    在一次loop中记录最后一次交换的位置,那么该位置后面数一定都是排好了的。所以这一部分就不用参与下面的比较了。

过程:

  1. 比较相邻的元素。如果第一个比第二个大,就交换他们两个。
  2. 对每一对相邻元素作同样的工作,从开始第一对到结尾的最后一对。在这一点,最后的元素应该会是最大的数。
  3. 针对所有的元素重复以上的步骤,除了最后一个。
  4. 持续每次对越来越少的元素重复上面的步骤,直到没有任何一对数字需要比较
    ------百度百科
def bubble_sort(nums):
    for i in range(1, len(nums)):
        for j in range(0, len(nums)-i):
            if nums[j+1] < nums[j]:
                nums[j+1], nums[j] = nums[j], nums[j+1]

a = [9,3,2,7,1,8,0,4,10,6]
bubble_sort(a)
print(a)

6. 鸡尾酒排序/双向冒泡排序

  1. 空间复杂度为:O(1)
  2. 平均和最差的时间复杂度为O(n^2),但如果序列在一开始已经大部分排序过的话,趋近于O(n)。
    原因:双向排序可以避免大量小的数在最右边而小数移动缓慢的情况(升序)

过程:

  1. 传统冒泡气泡排序的双向进行,先让气泡排序由左向右进行
  2. 再来让气泡排序由右往左进行,如此完成一次排序的动作
  3. 使用left与right两个旗标来记录左右两端已排序的元素位置:
    当left = right时:结束
    否则:继续1
def bubble_sort(nums):
    left = 1
    right = len(nums)
    while left < right:
        for i in range(0, len(nums)-left):
            if nums[i+1] < nums[i]:
                nums[i+1], nums[i] = nums[i], nums[i+1]
        left += 1
        for j in range(len(nums)-1, len(nums)-right, -1):
            if nums[j] < nums[j-1]:
                nums[j], nums[j-1] = nums[j-1], nums[j]
        right -= 1
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 202,723评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,080评论 2 379
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 149,604评论 0 335
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,440评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,431评论 5 364
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,499评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,893评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,541评论 0 256
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,751评论 1 296
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,547评论 2 319
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,619评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,320评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,890评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,896评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,137评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,796评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,335评论 2 342

推荐阅读更多精彩内容

  • 概述:排序有内部排序和外部排序,内部排序是数据记录在内存中进行排序,而外部排序是因排序的数据很大,一次不能容纳全部...
    每天刷两次牙阅读 3,726评论 0 15
  • 概述 排序有内部排序和外部排序,内部排序是数据记录在内存中进行排序,而外部排序是因排序的数据很大,一次不能容纳全部...
    蚁前阅读 5,159评论 0 52
  • 概述 排序有内部排序和外部排序,内部排序是数据记录在内存中进行排序,而外部排序是因排序的数据很大,一次不能容纳全部...
    闲云清烟阅读 756评论 0 6
  • 一、 单项选择题(共71题) 对n个元素的序列进行冒泡排序时,最少的比较次数是( )。A. n ...
    貝影阅读 8,948评论 0 10
  • 自从看了周梵老师的书《当你开始爱自己全世界都会来爱你》,我就开始慢慢的自我观察、自我觉知。 其实我天生具有天蝎座的...
    唱妈阅读 669评论 0 3