前言

本文章是建立在插入排序的基础上写的喔,如果有对插入排序还有不懂的童鞋,可以看看这里。

直接/折半插入排序 2路插入排序


希尔排序解释

希尔排序 Shell Sort 又名"缩小增量排序",是对直接插入排序更加高效的改进版本。它是由 Donald Shell 于1959年提出的一种排序算法。

希尔排序 其原理是设置一个增量incre,在原序列上每隔一个增量选取一个数据元素,将这些选取的元素构造成一个子序列

每设一个增量,我们每次会得到一组子序列 (子序列个数和当前增量相等),然后分别对这些子序列进行 直接插入排序

随着增量的减少,重复上述的操作,直到增量incre1 时,最后完成整个序列的排序。


希尔排序增量的选取

原始希尔增量

对于 希尔排序 的增量的选取,Donald Shell 一开始提出增量每次为上次的 1/2

也就是说,若数组长度为n,一开始增量为 n/2,之后每次增量都取上次的 1/2。

Knuth序列

Knuth序列:以逆向形式从1开始,通过递归表达式 interval = 3 * interval + 1 来产生,以此来得到间隔大小。

由此我们可以得到如下的增量选取方式:

具体方法是若数组长度为n,一开始增量取 n/3 向下取整 + 1。然后每次都取上次的增量的 1/3向下取整 + 1

当n足够大时,使用 Knuth序列 得到的增量选取方式,可以一定程度上提高希尔排序的效率。

(上面说的东西其实我也不知道对不对,至于为什么这样取,我也不知道哇哇哇 )

(在后文的程序中,我会采取此方式选取增量。


希尔排序动态演示

我们以 [6,5,2,4,1,3] 为例进行动态演示

第一次取增量,构造一组子序列

第一次取增量后,对每个子序列进行直接插入排序

第二次取增量,构造一组子序列

第二次取增量后,对每个子序列进行直接插入排序

第三次取增量,构造一组子序列

第三次取增量后,对每个子序列进行直接插入排序


希尔排序时间复杂度

对于 希尔排序 的时间复杂度,我真的研究了好久,用尽自己毕生所学的高数知识。

但是能力有限,最终没有探索出什么结果呜呜呜(是我太菜了对不起)

不过有人通过大量的实验,给出了较好的结果:当n较大时,比较和移动的次数约在 n^1.25(1.6n)^1.25 之间。


希尔排序核心代码

//设置增量  每隔一个增量取一个数 组成长度相同的子序列
void ShellSort(vector<int> &v){
int n = v.size();
int incre = n; //初始化增量
while(incre > 1){ //最后一次增量为1
incre = incre / 3 + 1; //除三向下取整加一
//(至于为什么这样取, 哇咖喱嘛三)
for(int i = incre; i < n; i++){
int key = v[i]; //当前需要插入的数
int j = i - incre;
while(j >= 0 && v[j] > key){
v[j + incre] = v[j];
j -= incre; //对每个子序列进行直接插入排序
}
v[j + incre] = key; //插入到合适位置
}
}
}

完整程序源代码

#include<iostream>
#include<vector>
#include<time.h>
using namespace std; //设置增量 每隔一个增量取一个数 组成长度相同的子序列
void ShellSort(vector<int> &v){
int n = v.size();
int incre = n; //初始化增量
while(incre > 1){ //最后一次增量为1
incre = incre / 3 + 1; //除三向下取整加一
//(至于为什么这样取, 哇咖喱嘛三)
for(int i = incre; i < n; i++){
int key = v[i]; //当前需要插入的数
int j = i - incre;
while(j >= 0 && v[j] > key){
v[j + incre] = v[j];
j -= incre; //对每个子序列进行直接插入排序
}
v[j + incre] = key; //插入到合适位置
}
}
} void show(vector<int> &v){
for(auto &x : v)
cout<<x<<" ";
cout<<endl;
} main(){
vector<int> v;
int n = 50;
srand((int)time(0));
while(n--)
v.push_back(rand() % 100 + 1); show(v); ShellSort(v); cout<<endl<<endl;
show(v);
}

程序运行结果图

[排序算法] 希尔排序 (C++)的更多相关文章

  1. 排序算法--希尔排序(Shell Sort)_C#程序实现

    排序算法--希尔排序(Shell Sort)_C#程序实现 排序(Sort)是计算机程序设计中的一种重要操作,也是日常生活中经常遇到的问题.例如,字典中的单词是以字母的顺序排列,否则,使用起来非常困难 ...

  2. 数据结构和算法(Golang实现)(22)排序算法-希尔排序

    希尔排序 1959 年一个叫Donald L. Shell (March 1, 1924 – November 2, 2015)的美国人在Communications of the ACM 国际计算机 ...

  3. 使用 js 实现十大排序算法: 希尔排序

    使用 js 实现十大排序算法: 希尔排序 希尔排序 refs xgqfrms 2012-2020 www.cnblogs.com 发布文章使用:只允许注册用户才可以访问!

  4. C数据结构排序算法——希尔排序法用法总结(转http://www.cnblogs.com/skywang12345/p/3597597.html)

    希尔排序介绍 希尔排序(Shell Sort)是插入排序的一种,它是针对直接插入排序算法的改进.该方法又称缩小增量排序,因DL.Shell于1959年提出而得名. 希尔排序实质上是一种分组插入方法.它 ...

  5. Python排序算法——希尔排序(Shell’s Sort)

    有趣的事,Python永远不会缺席! 如需转发,请注明出处:小婷儿的python https://www.cnblogs.com/xxtalhr/p/10793487.html 一.希尔排序(Shel ...

  6. js 实现排序算法 -- 希尔排序(Shell Sort)

    原文: 十大经典排序算法(动图演示) 希尔排序 1959年Shell发明,第一个突破O(n2)的排序算法,是简单插入排序的改进版.它与插入排序的不同之处在于,它会优先比较距离较远的元素.希尔排序又叫缩 ...

  7. JavaScript排序算法——希尔排序

    <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/ ...

  8. 排序算法-希尔排序(Java)

    package com.rao.sort; import java.util.Arrays; /** * @author Srao * @className ShellSort * @date 201 ...

  9. Java排序算法——希尔排序

    package sort; //================================================= // File Name : ShellSort //------- ...

  10. 数据结构和算法(Golang实现)(20)排序算法-选择排序

    选择排序 选择排序,一般我们指的是简单选择排序,也可以叫直接选择排序,它不像冒泡排序一样相邻地交换元素,而是通过选择最小的元素,每轮迭代只需交换一次.虽然交换次数比冒泡少很多,但效率和冒泡排序一样的糟 ...

随机推荐

  1. BI系统的分布式部署原理和技术实现

    1.什么是分布式 关于"分布式系统"的定义,我们先看下书中是怎么说的.<分布式系统原理和范型>一书中是这样定义分布式系统的:"分布式系统是若干独立计算机的集合 ...

  2. C/C++ Capstone 引擎源码编译

    Capstone 是一个轻量级的多平台.多架构的反汇编框架.Capstone 旨在成为安全社区中二进制分析和反汇编的终极反汇编引擎.Capstone的编译非常简单只需要一步即可轻松得到对应的Lib库文 ...

  3. 国产PLM软件在创新实践中强势崛起

    近日,"璞华PLM"先后获得微度医疗.埃特斯等多个客户的订单,即使在疫情环境下也展现出了强劲的高速增长.在产品生命周期管理(PLM,Product Lifecycle Manage ...

  4. HBuilder X之uniapp最适合的代码补全模板

    { // 注意:本文档仅支持单行注释,并且'//'前不能有任何非空字符!!! // // HBuilderX使用json扩展代码块,兼容vscode的代码块格式 // 本文档修改完毕,保存即可生效,无 ...

  5. Fluentd 使用 multiline 解析器来处理多行日志

    转载自:https://mp.weixin.qq.com/s?__biz=MzU4MjQ0MTU4Ng==&mid=2247500439&idx=1&sn=45e9e0e0ef ...

  6. Kubernetes 调度 - 污点和容忍度详解

    当我们使用节点亲和力(Pod 的一个属性)时,它会将Pod吸引到一组节点(作为偏好或硬性要求).污点的行为完全相反,它们允许一个节点排斥一组 Pod. 在 Kubernetes 中,您可以标记(污染) ...

  7. 使用nginx部署Django静态文件配置

    首先,我们配置静态文件,要在setting.py里面加入如下几行代码: # settings.py # the settings above # STATIC SETTINGS STATIC_URL ...

  8. Skywalking Swck Agent注入实现分析

    项目地址: GitHub - apache/skywalking-swck: Apache SkyWalking Cloud on Kubernetes 项目简介: A bridge project ...

  9. 使用@Param标识参数

    可以通过@Param注解标识mapper接口中的方法参数 此时,会将这些参数放在map集合中,以@Param注解的value属性值为键,以参数为值: 以 param1,param2...为键,以参数为 ...

  10. 驱动开发:内核枚举LoadImage映像回调

    在笔者之前的文章<驱动开发:内核特征码搜索函数封装>中我们封装实现了特征码定位功能,本章将继续使用该功能,本次我们需要枚举内核LoadImage映像回调,在Win64环境下我们可以设置一个 ...