python编程实现希尔排序


Posted in Python onApril 13, 2017

观察一下”插入排序“:其实不难发现她有个缺点:

如果当数据是”5, 4, 3, 2, 1“的时候,此时我们将“无序块”中的记录插入到“有序块”时,估计俺们要崩盘,每次插入都要移动位置,此时插入排序的效率可想而知。  

shell根据这个弱点进行了算法改进,融入了一种叫做“缩小增量排序法”的思想,其实也蛮简单的,不过有点注意的就是:

增量不是乱取,而是有规律可循的。

python编程实现希尔排序

希尔排序时效分析很难,关键码的比较次数与记录移动次数依赖于增量因子序列d的选取,特定情况下可以准确估算出关键码的比较次数和记录的移动次数。目前还没有人给出选取最好的增量因子序列的方法。增量因子序列可以有各种取法,有取奇数的,也有取质数的,但需要注意:增量因子中除1 外没有公因子,且最后一个增量因子必须为1。希尔排序方法是一个不稳定的排序方法。

首先要明确一下增量的取法(这里图片是copy别人博客的,增量是奇数,我下面的编程用的是偶数):

      第一次增量的取法为: d=count/2;

      第二次增量的取法为:  d=(count/2)/2;

      最后一直到: d=1; 

好,注意看图了,第一趟的增量d1=5, 将10个待排记录分为5个子序列,分别进行直接插入排序,结果为(13, 27, 49, 55, 04, 49, 38, 65, 97, 76)

第二趟的增量d2=3, 将10个待排记录分为3个子序列,分别进行直接插入排序,结果为(13, 04, 49, 38, 27, 49, 55, 65, 97, 76)

第三趟的增量d3=1, 对整个序列进行直接插入排序,最后结果为(04, 13, 27, 38, 49, 49, 55, 65, 76, 97)

重点来了。当增量减小到1时,此时序列已基本有序,希尔排序的最后一趟就是接近最好情况的直接插入排序。可将前面各趟的"宏观"调整看成是最后一趟的预处理,比只做一次直接插入排序效率更高。

本人是学python的,今天用python实现了希尔排序。

def ShellInsetSort(array, len_array, dk): # 直接插入排序
 for i in range(dk, len_array): # 从下标为dk的数进行插入排序
 position = i
 current_val = array[position] # 要插入的数

 index = i
 j = int(index / dk) # index与dk的商
 index = index - j * dk

 # while True: # 找到第一个的下标,在增量为dk中,第一个的下标index必然 0<=index<dk
 # index = index - dk
 # if 0<=index and index <dk:
 # break


 # position>index,要插入的数的下标必须得大于第一个下标
 while position > index and current_val < array[position-dk]:
 array[position] = array[position-dk] # 往后移动
 position = position-dk
 else:
 array[position] = current_val



def ShellSort(array, len_array): # 希尔排序
 dk = int(len_array/2) # 增量
 while(dk >= 1):
 ShellInsetSort(array, len_array, dk)
 print(">>:",array)
 dk = int(dk/2)

if __name__ == "__main__":
 array = [49, 38, 65, 97, 76, 13, 27, 49, 55, 4]
 print(">:", array)
 ShellSort(array, len(array))

输出:

>: [49, 38, 65, 97, 76, 13, 27, 49, 55, 4]
>>: [13, 27, 49, 55, 4, 49, 38, 65, 97, 76]
>>: [4, 27, 13, 49, 38, 55, 49, 65, 97, 76]
>>: [4, 13, 27, 38, 49, 49, 55, 65, 76, 97]

首先你得先会插入排序,不会你必然看不懂。 

python编程实现希尔排序

插入排序,即是对上图三个黄色框中的数进行插入排序。举个例子:13,55,38,76

直接看55,55<13, 不用移动。接着看38,38<55,那么55后移,数据变为[13,55,55,76],接着比较13<38, 那么38替换55,变成[13,38,55,76]。其它同理,略。

这里有个问题,比如第二个黄色框[27,4,65],4<27, 那27往后移,接着4就替换第一个,数据变成[4,27,65],但是计算机怎么知道4就是在第一个啊??

我的做法是,先找出[27,4,65]第一个数的下标,在这个例子中27的下标为1。当要插入的数的下标大于第一个下标1时,才可以往后移,前一个数不可以往后移有两种情况,一种是前面有数据,且小于要插入的数,那你只能插在它后面。另一种,很重要,当要插入数比前面所有数都小时,那插入数肯定是放在第一个,此时要插入数的下标=第一个数的下标。(这段话,感觉初学者应该不大懂……)

为了找到第一个数的下标,最开始想的是用循环,一直到最前面:

while True: # 找到第一个的下标,在增量为dk中,第一个的下标index必然 0<=index<dk
 index = index - dk
 if 0<=index and index <dk:
 break

在Debug时,发现用循环太浪费时间了,特别是当增量d=1时,直接插入排序为了插入列表最后一个数,得循环减1,直到第一个数的下标,后来我学聪明了,用下面的方法:

j = int(index / dk) # index与dk的商
index = index - j * dk

时间复杂度:

希尔排序的时间复杂度是所取增量序列的函数,尚难准确分析。有文献指出,当增量序列为d[k]=2^(t-k+1)时,希尔排序的时间复杂度为O(n^1.5), 其中t为排序趟数。

稳定性: 不稳定

希尔排序效果: 

 python编程实现希尔排序

参考资料: 编程是我自己实现的。建议Debug看看运行过程

1.非系统的学习也是在浪费时间 2.做一个会欣赏美,懂艺术,会艺术的技术人

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Python 相关文章推荐
Python实现的从右到左字符串替换方法示例
Jul 06 Python
python3的输入方式及多组输入方法
Oct 17 Python
Python 实现两个列表里元素对应相乘的方法
Nov 14 Python
pycharm+PyQt5+python最新开发环境配置(踩坑)
Feb 11 Python
详解python使用turtle库来画一朵花
Mar 21 Python
django框架模型层功能、组成与用法分析
Jul 30 Python
Python实现上下文管理器的方法
Aug 07 Python
pycharm2020.2 配置使用的方法详解
Sep 16 Python
Python实现给PDF添加水印的方法
Jan 25 Python
浅谈pytorch中stack和cat的及to_tensor的坑
May 20 Python
如何在pycharm中快捷安装pip命令(如pygame)
May 31 Python
python flappy bird小游戏分步实现流程
Feb 15 Python
python实现解数独程序代码
Apr 12 #Python
python访问抓取网页常用命令总结
Apr 11 #Python
Python 登录网站详解及实例
Apr 11 #Python
python实现FTP服务器服务的方法
Apr 11 #Python
python读写json文件的简单实现
Apr 11 #Python
Python实现Mysql数据库连接池实例详解
Apr 11 #Python
详解Python中类的定义与使用
Apr 11 #Python
You might like
CI框架(CodeIgniter)操作redis的方法详解
2018/01/25 PHP
使用PHPExcel导出Excel表
2018/09/08 PHP
for 循环性能比较 提高for循环的效率
2009/03/19 Javascript
jQuery 注意事项 与原因分析
2009/04/24 Javascript
关于COOKIE个数与大小的问题
2011/01/17 Javascript
基于JQuery 的消息提示框效果代码
2011/07/31 Javascript
javascript unicode与GBK2312(中文)编码转换方法
2013/11/14 Javascript
JQuery下拉框应用示例介绍
2014/04/23 Javascript
js实现上传图片预览的方法
2015/02/09 Javascript
javascript函数特点实例分析
2015/05/14 Javascript
Ionic 2 实现列表滑动删除按钮的方法
2017/01/22 Javascript
JS中获取 DOM 元素的绝对位置实例详解
2018/04/23 Javascript
Angular Material Icon使用详解
2018/11/07 Javascript
js根据需要计算数组中重复出现某个元素的个数
2019/01/18 Javascript
PHP实现基于Redis的MessageQueue队列封装操作示例
2019/02/02 Javascript
微信小程序实现购物车代码实例详解
2019/08/29 Javascript
JavaScript 中判断变量是否为数字的示例代码
2020/10/22 Javascript
[00:36]DOTA2风云人物相约完美“圣”典 12月17日不见不散
2016/11/30 DOTA
python中urllib模块用法实例详解
2014/11/19 Python
Python自动重试HTTP连接装饰器
2015/04/28 Python
Python中staticmethod和classmethod的作用与区别
2018/10/11 Python
Python 3.x基于Xml数据的Http请求方法
2018/12/28 Python
Python如何用filter函数筛选数据
2020/03/05 Python
python模拟斗地主发牌
2020/04/22 Python
Python正则表达式如何匹配中文
2020/05/27 Python
VSCode中autopep8无法运行问题解决方案(提示Error: Command failed,usage)
2021/03/02 Python
Waterford加拿大官方网站:世界著名的水晶杯品牌
2016/11/01 全球购物
英国最大的宝石首饰超市:QP Jewellers
2018/09/23 全球购物
美国亚洲时尚和美容产品的一站式网上商店:Stylevana
2019/09/05 全球购物
《钱学森》听课反思
2014/03/01 职场文书
项目工作说明书
2014/07/29 职场文书
2014年社区国庆节活动方案
2014/09/16 职场文书
golang slice元素去重操作
2021/04/30 Golang
Python办公自动化之教你如何用Python将任意文件转为PDF格式
2021/06/28 Python
golang实现浏览器导出excel文件功能
2022/03/25 Golang
python 闭包函数详细介绍
2022/04/19 Python