「数组」希尔排序 / 区间增量优化(C++)

目录

概述

思路

核心概念:增量d

算法过程

流程

Code

优化方案

区间增量优化

Code(pro)

复杂度


概述

我们在「数组」冒泡排序|选择排序|插入排序 / 及优化方案(C++)中讲解了插入排序。

它有这么两个特点:

①待排序元素较少时效率高。

②待排序元素较有序时效率高。

正如同快速排序时冒泡排序的究极promax进化版,希尔排序则是充分利用了这两个特点的插入排序promax进化版。


思路

战略是这样的:多次进行小数目的插入排序使得数组变得相对有序。

我们要采取一点策略:

通过多轮小型插入排序使得数组逐渐有序,然后就可以将小型插入排序变成中型插入排序。通过多轮中型插入排序使得数组几乎有序,然后就可以将小型插入排序变成整体插入排序。

通过一轮整体插入排序使得数组完全有序。

这个“小型的插入排序”的目的是使得数组逐渐有序,这意味这我们要在整个数组中挑选几个数出来,对他们进行插入排序。

这种挑选是很有讲究的:

我们挑选的数必须能均等地位于整个数组的不同位置中,这样才能使整个数组愈发有序。

我们挑选的数必须能覆盖整个数组,这样才能使整个数组整体愈发有序。

于是就有了增量的概念。


核心概念:增量d

增量d的本质就是对整个数组进行间隔分组:

我们将arr[i],arr[i+d],arr[i+2d]...分为一组,在组内进行插入排序。

完成一组后再完成下一组,直到所有组都进行了组内插入排序。之后减小增量d重新分组,重复上述过程,直到d=1,进行完整的插入排序。

通常我们初始化d=len/2,然后依次d/=2。(向下取整)

例如:

 len=11;
 arr[i] 7 1 8 9 5 6 4 2 3 10 0
                      ↓d=len/2;
┌--------------------------------------------┐
 d=5;
     i  0 1 2 3 4 5 6 7 8 9 10
 arr[i] 7 1 8 9 5 6 4 2 3 10 0
               ↓d↓
 group0 7----d----6          0
 group1   1----d----4
 group2     8----d----2
 group3       9----d----3
 group4         5----d----10
        ↓insertion_sort()↓
 group0 0         6          7
 group1   1         4
 group2     2         8
 group3       3         9
 group4         5         10
          ↓after sorted↓
 arr[i] 0 1 2 3 5 6 4 8 9 10 7
└--------------------------------------------┘
                      ↓d/=2;
┌--------------------------------------------┐
 d=2;
     i  0 1 2 3 4 5 6 7 8 9 10
 arr[i] 0 1 2 3 5 6 4 8 9 10 7
               ↓d↓
 group0 0-d-2   5   4   9    7
 group1   1-d-3   6   8   10
        ↓insertion_sort()↓
 group0 0   2   4   5   7    9
 group1   1   3   6   8   10
          ↓after sorted↓
 arr[i] 0 1 2 3 4 6 5 8 7 10 9
└--------------------------------------------┘
                      ↓d/=2;
┌--------------------------------------------┐
 d=1;
     i  0 1 2 3 4 5 6 7 8 9 10
 arr[i] 0 1 2 3 4 6 5 8 7 10 9
        ↓insertion_sort()↓
 arr[i] 0 1 2 3 4 5 6 7 8 9 10
└--------------------------------------------┘

我们注意到,d的值和分组数量是相等的

因为arr[i]与arr[i+d]为同组,而arr[i+d]与arr[i]间共有d-1组各不相同,再加上arr[i]这一组,共d组。

这一点将会在分组代码实现时利用到。 

*注意*:分组图只是我们的具象化表达,希尔排序是原地算法,不会使用额外的空间储存每一组。 


算法过程

流程

共有四层循环:

①最外层循环(增量减半缩小层)while (d/=2)控制增量减半

②次外层循环(按照增量分组层)for (int group = 0; group < d; group++)进行分组

③次内层循环for (int i = group+d; i < len; i += d)进行组内插入排序(根据插入排序的原理,首个元素可以跳过)

④最内层循环for ( j= i-d; j >= 0; j -= d)将组内的元素插入到组内的有序区中。

你会发现内部的两层循环就是普通插入排序的是实现,只不过普通插入排序的增量d始终为1。

Code

void shell_sort(int arr[], int len) {
	int d = len;
	while (d /= 2) {
		for (int group = 0; group < d; group++) {
			for (int i = group+d; i < len; i += d) {
				int temp = arr[i], j = i - d;
				for (; j >= 0; j -= d) {
					if (temp < arr[j])arr[j + d] = arr[j];
					else break;
				}
				arr[j + d] = temp;
			}
		}
	}
}

优化方案

区间增量优化

Knuth大神提出了另一种增量策略:d=d/3+1。(+1是为了使得d==2时下次取到d==1)

你会意识到上一种分组的增量减半缩小层是log₂N级别的,而这种则是log₃N级别的

但是这种优化不一定是最理想的,其实与上一种分组各有胜负:

因为这只是优化了增量减半缩小层,而每层内部进行了更多的比较。

Code(pro)

void SLsort(int arr[], int len) {
	int d = len;
	while (d = d/3+1) {
		for (int group = 0; group < d; group++) {
			for (int i = group + d; i < len; i += d) {
				int temp = arr[i], j = i - d;
				for (; j >= 0; j -= d) {
					if (temp < arr[j])arr[j + d] = arr[j];
					else break;
				}
				arr[j + d] = temp;
			}
		}
		if (d == 1)break;
	}
}

*注意*: 需要加入d==1的判断语句来结束最外层循环。


复杂度

时间复杂度:O(n¹·³)(或:O(nlog²n)

空间复杂度:O(1)

事实上,希尔排序的时间复杂度不是nlogn,它的证明极其困难,略去不表。

百万数量级抗压测试

int main()
{   int nums = 5000000;
	int* arr1 = new int[nums];
	int* arr2 = new int[nums];
 	for (int i = 0; i < nums; i++) {
		int x = mt()%1000;
		arr1[i] =arr2[i]= x;
	}

	DWORD tick1 = GetTickCount64();
	shell_sort(arr1, nums);//show(arr, nums);
	DWORD tick2 = GetTickCount64();
	cout <<"Shell's strategy(ms):" << tick2 - tick1 << endl;

	DWORD tick3 = GetTickCount64();
	SLsort(arr2, nums);//show(arr, nums);
	DWORD tick4 = GetTickCount64();
	cout <<"Knuth's strategy(ms):" << tick4 - tick3 << endl;

	delete[] arr1;
	delete[] arr2;
	return 0;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/871500.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

路由偏好详解

路由偏好对网络性能和数据传输效率有着重要影响。本文将从路由偏好的相关概念、影响因素和实际应用&#xff0c;同时结合IP数据云的功能展示其在优化路由选择中的作用。 路由偏好指网络设备在选择路由路径时所倾向的特定策略或条件。它基于多种因素进行决策&#xff0c;例如网络…

CSS继承、盒子模型、float浮动、定位、diaplay

一、CSS继承 1.文字相关的样式会被子元素继承。 2.布局样式相关的不会被子元素继承。&#xff08;用inherit可以强行继承&#xff09; 实现效果&#xff1a; 二、盒子模型 每个标签都有一个盒子模型&#xff0c;有内容区、内边距、边框、外边距。 从内到外&#xff1a;cont…

矩阵中的最大得分(Lc3148)——动态规划

给你一个由 正整数 组成、大小为 m x n 的矩阵 grid。你可以从矩阵中的任一单元格移动到另一个位于正下方或正右侧的任意单元格&#xff08;不必相邻&#xff09;。从值为 c1 的单元格移动到值为 c2 的单元格的得分为 c2 - c1 。 你可以从 任一 单元格开始&#xff0c;并且必须…

Maven的简单使用

Maven使用 Maven的作用1. 自动构建标准化的java项目结构(1) 项目结构① 约定目录结构的意义② 约定大于配置 (2)项目创建坐标坐标的命名方法&#xff08;约定&#xff09; 2. 帮助管理java中jar包的依赖(1) 配置使用依赖引入属性配置 (2) maven指令(3) 依赖的范围(4) 依赖传递(…

ChatGPT 3.5/4.0 新手使用手册(详细版)

1. 什么是 ChatGPT&#xff1f; ChatGPT是由 OpenAI 开发的先进人工智能语言模型&#xff0c;能够理解并生成自然语言文本。它可以帮助你进行写作、回答问题、提供建议&#xff0c;甚至参与对话。ChatGPT 3.5 和 4.0 是两个不同版本&#xff0c;它们都拥有强大的语言处理能力&…

24/8/17算法笔记 策略梯度reinforce算法

import gym from matplotlib import pyplot as plt %matplotlib inline#创建环境 env gym.make(CartPole-v0) env.reset()#打印游戏 def show():plt.imshow(env.render(mode rgb_array))plt.show() show()定义网络模型 import torch #定义模型 model torch.nn.Sequential(t…

算法日记day 44(动归之编辑距离|回文字串|最长回文子序列)

一、编辑距离 题目&#xff1a; 给你两个单词 word1 和 word2&#xff0c; 请返回将 word1 转换成 word2 所使用的最少操作数 。 你可以对一个单词进行如下三种操作&#xff1a; 插入一个字符删除一个字符替换一个字符 示例 1&#xff1a; 输入&#xff1a;word1 "…

仿照ContentLoadingProgressBar 的特点在Android项目中自定义Loading对话框

ContentLoadingProgressBar 是 Android 中的一个控件&#xff0c;继承自 ProgressBar。它在 ProgressBar 的基础上添加了一些特殊功能&#xff0c;主要用于在加载内容时显示进度。它的一些主要特点如下&#xff1a; 自动隐藏和显示&#xff1a;ContentLoadingProgressBar 会在…

初级python代码编程学习----简单的图形化闹钟小程序

我们来创建一个简单的图形化闹钟程序通常需要使用图形用户界面&#xff08;GUI&#xff09;库。以下是使用Python的Tkinter库创建一个基本闹钟程序的步骤&#xff1a; 环境准备 确保已安装Python。安装Tkinter库&#xff08;Python 3.8及以上版本自带Tkinter&#xff0c;无需…

软件测试学习笔记丨Allure2报告添加附件报告定制

本文转自测试人社区&#xff0c;原文链接&#xff1a;https://ceshiren.com/t/topic/31810 一、Allure2报告中添加附件-图片 1.1 附件类型 TEXT ("text/plain", "txt") CSV ("text/csv", "csv") TSV ("text/tab-separated-v…

leetcode:2119. 反转两次的数字(python3解法)

难度&#xff1a;简单 反转 一个整数意味着倒置它的所有位。 例如&#xff0c;反转 2021 得到 1202 。反转 12300 得到 321 &#xff0c;不保留前导零 。 给你一个整数 num &#xff0c;反转 num 得到 reversed1 &#xff0c;接着反转 reversed1 得到 reversed2 。如果 reverse…

GEC6818开发板的学习

1、开发板的简介 首先连接 开发板与电脑,需电脑安装串口驱动:例CH340 2、开发板的特性: 像素:800*480Pix分辨率:高,宽两个维度的像素点数目开发板色深为32位一个像素点占4个字节:分别为灰度保留位、RGB三原色各占一位3、为什么要内存映射 虽然LCD设备本质上也可以看作…

OW-VISCap——开放世界视频实例分割方法研究

概述 论文地址&#xff1a;https://arxiv.org/pdf/2404.03657 本文提出了一种名为 OW-VISCap&#xff08;开放世界视频实例分割和字幕&#xff09;的方法。其三大贡献是 开放世界对象查询&#xff1a;除了已知对象查询外&#xff0c;还引入了开放世界对象查询&#xff0c;以发…

【安全靶场】-DC-7

❤️博客主页&#xff1a; iknow181 &#x1f525;系列专栏&#xff1a; 网络安全、 Python、JavaSE、JavaWeb、CCNP &#x1f389;欢迎大家点赞&#x1f44d;收藏⭐评论✍ 一、收集信息 1.查看主机是否存活 nmap -T4 -sP 192.168.216.149 2.主动扫描 看开放了哪些端口和功能 n…

WPF调用CEF插件运行时启动CefSharp.BrowserSubprocess.exe三个进程

cefsharp.browsersubprocess.exe 是CefSharp&#xff08;一个基于Chromium的开源浏览器控件&#xff09;的一部分。这个可执行文件通常在以下情况下启动&#xff1a; 渲染进程&#xff1a;CefSharp使用多进程架构&#xff0c;类似于Chrome浏览器。cefsharp.browsersubprocess.e…

ArcGIS 数据服务在三维 Cesium/SuperMap 项目中使用遇到的一些问题及其解决方法

ArcGIS 数据服务在三维 Cesium/SuperMap 项目中使用遇到的一些问题及其解决方法 一、三维系统支持的 ArcGIS 服务及其投影 1、动态服务 ArcGIS 动态服务的数据&#xff0c;支持任意投影在三维系统中加载。 2、切片服务 ArcGIS 切片服务仅支持 3857(web 墨卡托投影)&#x…

19529 照明灯安装

### 详细分析 这个问题可以通过二分查找和贪心算法来解决。我们需要找到一个最大值&#xff0c;使得在这个最大值下&#xff0c;能够在给定的坐标上安装 k 个照明灯&#xff0c;并且相邻的照明灯之间的距离至少为这个最大值。 ### 思路 1. **排序**&#xff1a;首先对给定的…

arthas源码刨析:启动 (1)

文章目录 arthas-bootBootstrap Created with Raphal 2.3.0 开始 检查监听端口 jps 列表java应用 下载 lib 依赖 功能移交给 arthas-core 结束 arthas-boot 该module 的代码只有3个类&#xff1a; Bootstrap 启动类 Bootstrap &#xff0c;开头的注解就是 alibaba 的 cli 中…

凡图公益行:凡图家庭教育以行动筑梦,点亮孩子心中的光芒

在教育的路上&#xff0c;每一步都承载着未来的希望&#xff0c;凡图(山东)教育科技集团有限公司一直致力于为每一个孩子及家庭提供最优质的心理咨询服务。 在这样的背景下&#xff0c;凡图家庭教育以独特的使命感和责任感&#xff0c;成为了众多家庭信赖的伙伴。 也因此成为…

阿里Qwen2开源大模型本地部署及调试全攻略

阿里Qwen2开源大模型本地部署及调试全攻略 #Qwen2系列大模型性能卓越&#xff0c;超越业界知名模型。开源后受到AI开发者关注&#xff0c;支持多种语言&#xff0c;提升多语言理解。在预训练和微调上优化&#xff0c;实现智能水平提升。Qwen2系列模型在各项能力上均领先&#…