排序算法

下面算法均是使用Python实现:

插入排序

原理:在要排序的一组数中,假定前n-1个数已经排好序,现在将第n个数插到前面的有序数列中,使得这n个数也是排好顺序的。如此反复循环,直到全部排好顺序。通常使用在长度较小的数组的情况以及作为其它复杂排序算法的一部分,比如mergesort或quicksort。时间复杂度为 O(n^2) 。

  1. # 1nd: 两两交换
    def insertion_sort(seq):
       for i in range(1, len(seq)):
           j = i
           while j >= 0 and seq[j-1] > seq[j]:
               seq[j], seq[j-1] = seq[j-1], seq[j]
               j -= 1
       return seq


    # 2nd: 交换,最后处理没交换的
    def insertion_sort2(seq):
       for i in range(1, len(seq)):
           j = i-1
           key = seq[i]
           while j >= 0 and seq[j] > key:
               seq[j+1] = seq[j]
               j -= 1
           seq[j+1] = key
       return seq


    # 3nd: 加速版本,利用已经排好了序的进行二分查找
    def insertion_sort3(seq):
       for i in range(1, len(seq)):
           key = seq[i]
           # invariant: ``seq[:i]`` is sorted
           # find the least `low' such that ``seq[low]`` is not less then `key'.
           #   Binary search in sorted sequence ``seq[low:up]``:
           low, up = 0, i
           while up > low:
               middle = (low + up) // 2
               if seq[middle] < key:
                   low = middle + 1
               else:
                   up = middle
           # insert key at position ``low``
           seq[:] = seq[:low] + [key] + seq[low:i] + seq[i + 1:]
       return seq


    # 4nd: 原理同上,使用bisect,面试不推荐使用
    import bisect


    def insertion_sort4(seq):
       for i in range(1, len(seq)):
           bisect.insort(seq, seq.pop(i), 0, i)  # 默认插在相同元素的左边

       return seq

选择排序

原理:每一趟都选择当前后面最小的值和当前下标的值进行交换,适用在大型的数组,时间复杂度为 O(n^2)

  1. # 1nd: for
    def select_sort(seq):
       for i in range(0, len(seq)):
           mi = i
           # 选择坐标i后的最小值
           for j in range(i, len(seq)):
               if seq[j] < seq[mi]:
                   mi = j
           seq[mi], seq[i] = seq[i], seq[mi]
       return seq


    # 2nd: min
    def select_sort2(seq):
       for i, x in enumerate(seq):
           mi = min(range(i, len(seq)), key=seq.__getitem__)
           seq[i], seq[mi] = seq[mi], x
       return seq

冒泡排序

原理:比较数组中两两相邻的数,如果前者大于后者,就进行交换,重复地走访过要排序的数列,达到将最小的值移动到最上面的目的,适用于小型数组,时间复杂度为O(n^2)

  1. def bubble_sort(seq):
       for i in range(len(seq)-1):
           j = len(seq)-1
           while j > i:
               if seq[j] < seq[j-1]:
                   seq[j], seq[j-1] = seq[j-1], seq[j]
               j -= 1
       return seq


    def bubble_sort2(seq):
       for i in range(0, len(seq)):
           for j in range(i + 1, len(seq)):
               if seq[i] > seq[j]:
                   seq[i], seq[j] = seq[j], seq[i]
       return seq

快速排序

原理:从数组中选择pivot,分成两个数组,一个是比pivot小,一个是比pivot大,最后将这两个数组和pivot进行合并,最好情况下时间复杂度为O(n log n),最差情况下时间复杂度为O(n^2)

  1. def quick_sort(seq):
       if len(seq) <= 1:
           return seq

       pivot_idx = len(seq) // 2  # 将中间作为基准
       small, large = [], []
       for idx, val in enumerate(seq):
           if idx != pivot_idx:
               if val <= seq[pivot_idx]:
                   small.append(val)
               else:
                   large.append(val)
       small = quick_sort(small)
       large = quick_sort(large)
       return small + [seq[pivot_idx]] + large
     
    def quick_sort(seq, left, right):
       """
      单纯移动下标的实现
      :param seq:
      :param left:
      :param right:
      :return:
      """
       if left >= right:
           return seq

       # 选择参考点,该调整范围的第1个值
       target = seq[left]
       i, j = left, right

       # 循环判断直到遍历全部
       while i < j:
           # 从右边开始查找大于参考点的值
           while i < j and seq[j] >= target:
               j -= 1
           seq[i] = seq[j]  # 这个位置的值先挪到左边

           # 从左边开始查找小于参考点的值
           while i < j and seq[i] <= target:
               i += 1
           seq[j] = seq[i]  # 这个位置的值挪到右边

       # 写回改成的值
       seq[i] = target
       # 递归,并返回结果
       quick_sort(seq, left, i - 1)  # 递归左边部分
       quick_sort(seq, i + 1, right)  # 递归右边部分
       return seq

归并排序

原理:归并排序是建立在归并操作上的一种有效的排序算法,该算法是采用分治法(Divide and Conquer)的一个非常典型的应用。将已有序的子序列合并,得到完全有序的序列;即先使每个子序列有序,再使子序列段间有序。若将两个有序表合并成一个有序表,称为二路归并。

  1. # 1nd: 将两个有序数组合并到一个数组
    def merge(left, right):
       i, j = 0, 0
       result = []
       while i < len(left) and j < len(right):
           if left[i] <= right[j]:
               result.append(left[i])
               i += 1
           else:
               result.append(right[j])
               j += 1
       result += left[i:]
       result += right[j:]
       return result


    def merge_sort(seq):
       if len(seq) <= 1:
           return seq

       mid = len(seq) // 2
       left = merge_sort(seq[:mid])
       right = merge_sort(seq[mid:])
       return merge(left, right)


    # 2nd: use merge,面试不推荐使用
    from heapq import merge


    def merge_sort2(m):
       if len(m) <= 1:
           return m

       middle = len(m) // 2
       left = m[:middle]
       right = m[middle:]

       left = merge_sort(left)
       right = merge_sort(right)
       return list(merge(left, right))

堆排序

原理:堆排序(Heapsort)是指利用堆积树(堆)这种数据结构所设计的一种排序算法,是选择排序的一种。可以利用数组的特点快速定位指定索引的元素。堆分为大根堆和小根堆,是完全二叉树。大根堆的要求是每个节点的值都不大于其父节点的值。一般升序采用大顶堆,降序采用小顶堆。平均时间复杂度为O(n logn)。

  1. def heapify(seq, start, end):
       """
      找出从start到end的范围内的最大值,放在堆顶的位置
      :param seq:
      :param start:
      :param end:
      :return:
      """
       # start结点的左右子结点
       left, right = 2 * start + 1, 2 * (start + 1)
       ma = start
       # 选出最大值
       if left < end and seq[start] < seq[left]:
           ma = left
       if right < end and seq[ma] < seq[right]:
           ma = right
       if ma != start:
        # 找到最大值后调整位置
           seq[start], seq[ma] = seq[ma], seq[start]
           heapify(seq, ma, end)


    def heap_sort(seq):
       start, end = len(seq) // 2 - 1, len(seq)
       # 创建堆,i从根结点开始调整
       for i in range(start, -1, -1):
           heapify(seq, i, end)

       # 调整堆,将堆顶元素与末尾元素进行交换,使末尾元素最大。
       # 然后继续调整堆,再将堆顶元素与末尾元素交换,得到第二大元素。如此反复进行交换、重建、交换
       for i in range(end - 1, 0, -1):
           seq[i], seq[0] = seq[0], seq[i]
           heapify(seq, 0, i)

       return seq


    # 2nd: use heapq
    import heapq


    def heap_sort2(seq):
       """ Implementation of heap sort """
       heapq.heapify(seq)
       return [heapq.heappop(seq) for _ in range(len(seq))]

希尔排序

原理:希尔排序(Shell Sort)是插入排序的一种。也称缩小增量排序,是直接插入排序算法的一种更高效的改进版本。希尔排序是非稳定排序算法。希尔排序是把记录按下标的一定增量分组,对每组使用直接插入排序算法排序;随着增量逐渐减少,每组包含的关键词越来越多,当增量减至1时,整个文件恰被分成一组,算法便终止。

  1. def shell_sort(seq):
       count = len(seq)
       step = 2
       group = count // step
       while group > 0:
           for i in range(0, group):
               j = i + group
               while j < count:
                   k = j - group
                   key = seq[j]
                   while k >= 0:
                       if seq[k] > key:
                           seq[k + group] = seq[k]
                           seq[k] = key
                       k -= group
                   j += group
           group //= step
       return seq

区别

Python 实现排序算法的更多相关文章

  1. Python之排序算法:快速排序与冒泡排序

    Python之排序算法:快速排序与冒泡排序 转载请注明源地址:http://www.cnblogs.com/funnyzpc/p/7828610.html 入坑(简称IT)这一行也有些年头了,但自老师 ...

  2. python实现排序算法 时间复杂度、稳定性分析 冒泡排序、选择排序、插入排序、希尔排序

    说到排序算法,就不得不提时间复杂度和稳定性! 其实一直对稳定性不是很理解,今天研究python实现排序算法的时候突然有了新的体会,一定要记录下来 稳定性: 稳定性指的是 当排序碰到两个相等数的时候,他 ...

  3. python常见排序算法解析

    python——常见排序算法解析   算法是程序员的灵魂. 下面的博文是我整理的感觉还不错的算法实现 原理的理解是最重要的,我会常回来看看,并坚持每天刷leetcode 本篇主要实现九(八)大排序算法 ...

  4. 第四百一十五节,python常用排序算法学习

    第四百一十五节,python常用排序算法学习 常用排序 名称 复杂度 说明 备注 冒泡排序Bubble Sort O(N*N) 将待排序的元素看作是竖着排列的“气泡”,较小的元素比较轻,从而要往上浮 ...

  5. Python实现排序算法之快速排序

    Python实现排序算法:快速排序.冒泡排序.插入排序.选择排序.堆排序.归并排序和希尔排序 Python实现快速排序 原理 首先选取任意一个数据(通常选取数组的第一个数)作为关键数据,然后将所有比它 ...

  6. python 经典排序算法

    python 经典排序算法 排序算法可以分为内部排序和外部排序,内部排序是数据记录在内存中进行排序,而外部排序是因排序的数据很大,一次不能容纳全部的排序记录,在排序过程中需要访问外存.常见的内部排序算 ...

  7. python——常见排序算法解析

    算法是程序员的灵魂. 下面的博文是我整理的感觉还不错的算法实现 原理的理解是最重要的,我会常回来看看,并坚持每天刷leetcode 本篇主要实现九(八)大排序算法,分别是冒泡排序,插入排序,选择排序, ...

  8. python之排序算法

    排序是每个语言都需要学会的,不管是c++.java还是python,套路都是类似的 python中也有自带的排序函数sort,直接使用也可 闲来无事写了几个排序算法,各不相同 1.每次遇到最小的数都交 ...

  9. Python 八大排序算法速度比较

    这篇文章并不是介绍排序算法原理的,纯粹是想比较一下各种排序算法在真实场景下的运行速度. 算法由 Python 实现,用到了一些语法糖,可能会和其他语言有些区别,仅当参考就好. 测试的数据是自动生成的, ...

随机推荐

  1. 如何相互转换逗号分隔的字符串和List

    将逗号分隔的字符串转换为List 方法 1: 利用JDK的Arrays类 [java] view plain copy   ico_fork.svg1.5 KB String str = " ...

  2. [编织消息框架][netty源码分析]11 UnpooledHeapByteBuf 与 ByteBufAllocator

    每种ByteBuf都有相应的分配器ByteBufAllocator,类似工厂模式.我们先学习UnpooledHeapByteBuf与其对应的分配器UnpooledByteBufAllocator 如何 ...

  3. Linux 安装依赖库

    ###安装依赖库###yum -y install rsync net-snmp syslog net-snmp-devel wget patch screen gcc gcc-c++ autocon ...

  4. HTTP通过请求和响应的交换达成通信

    1. 通过请求和响应的交换达成通信 首先我们来看一个HTTP请求报文: GET/index.htm HTTP/1.1 HOST : hacker.jp HTTP协议起始行开头的GET表示请求访问服务器 ...

  5. 仿网易新闻app下拉标签选择菜单

    仿网易新闻app下拉标签选择菜单 仿网易新闻app下拉标签选择菜单,长按拖动排序,点击增删标签控件 ##示例  ##EasyTagDragView的使用 在layout布局里添加:  

  6. Unity3D-Shader-实现X光效果

    [旧博客转移 - 2016年1月3日 16:40 ] 最近学习了一些Shader效果,打算把学到的知识总结一下,这篇讲一下这种轮廓发光的效果(如下图所示),也有一些地方管这个叫X光     1.原理 ...

  7. 软考 程序员 下午考题 c语言 笔记

    1. 数组名 是表示数组空间首地址的指针常量,程序中不允许对常量赋值. 如 int  a[];   a就是数组名,表示数组控件首地址的指针常量 a = 0;是错误的,不允许对指针常量赋值 &a ...

  8. 部署服务能在Dynamics CRM Online上使用吗?

    部署服务有些时候有用,改动一些参数不需要重启IIS,也不需要去数据库中更改.比如,系统默认设置一个仪表盘(Dashboard)最多6个组件,不能超过这个数量,通过部署服务是可以更改的.部署服务既可以通 ...

  9. 浅谈javascript中的call与apply方法

    call方法与apply方法都是为了改变函数体内部this的指向. call方法与apply方法,这二者的作用完全一样,只是接受参数的方式不太一样. apply()方法: Function.apply ...

  10. VUE 分页组件

    <!DOCTYPE html><html> <head> <meta charset="UTF-8"> <title>& ...