【算法】直接插入排序

news2025/1/22 16:57:31

文章目录

  • 概念
  • 实现过程
    • 时间复杂度和空间复杂度
    • 代码示例
  • 总结

概念

直接插入排序(Insertion Sort)是一种简单直观的排序算法,它的基本思想是通过构建有序的子序列,逐步将无序的元素插入到有序序列中,最终实现整体的排序。

具体来说,直接插入排序的过程如下:

将待排序的序列看作一个有序序列和一个无序序列。初始时有序序列只包含第一个元素,无序序列包含剩下的元素。
逐个取出无序序列中的元素,并将它与有序序列中的元素逐个比较。
找到合适的位置后,将待插入的元素插入到有序序列中,同时将该位置后面的元素依次后移一位。
继续处理下一个无序元素,直到所有的元素都插入完成。
这样,经过多轮的比较和插入操作,整个序列就会逐渐变得有序,最终完成排序。

直接插入排序算法的优点是实现简单,思路清晰,对于小规模的数据集排序效果较好。然而,由于它的时间复杂度为O(n^2),在处理大规模数据时可能不够高效,因此在实际应用中需要综合考虑其他排序算法。

实现过程

假设前面 n-1(其中 n>=2)个数已经是排好顺序的,现将第 n 个数插到前面已经排好的序列中,然后找到合适自己的位置,使得插入第n个数的这个序列也是排好顺序的。
按照此法对所有元素进行插入,直到整个序列排为有序的过程,称为插入排序。
从小到大的插入排序整个过程如图示:
第一轮: 从第二位置的 6 开始比较,比前面 7 小,交换位置。
在这里插入图片描述

第二轮: 第三位置的 9 比前一位置的 7 大,无需交换位置
在这里插入图片描述
第三轮: 第四位置的 3 比前一位置的 9 小交换位置,依次往前比较。
在这里插入图片描述
第四轮: 第五位置的 1 比前一位置的 9 小,交换位置,再依次往前比较。
在这里插入图片描述

时间复杂度和空间复杂度

时间复杂度分析:

在最好情况下,即待排序序列已经有序的情况下,直接插入排序的时间复杂度是最低的,为O(n)。因为每个元素只需与前一个元素进行比较一次,无需进行元素的后移操作。
在最坏情况下,即待排序序列逆序的情况下,直接插入排序的时间复杂度是最高的,为O(n^2)。因为每个元素都需要与有序序列中的所有元素进行比较,并可能进行大量的元素后移操作。
平均情况下,直接插入排序的时间复杂度同样为O(n^2)。虽然有部分元素在比较时可能会提前找到合适的位置,减少了后移操作的次数,但整体来说,平均情况仍需进行大量的比较和后移操作。
空间复杂度分析:

直接插入排序的空间复杂度是O(1),即算法所需的额外存储空间是固定的,不随待排序序列规模的增长而增加。只需要使用常数级别的额外空间来保存一些临时变量。
稳定性:

直接插入排序是一种稳定的排序算法。当两个相等的元素进行比较时,如果当前元素小于等于前一个元素,则它们的相对顺序不会改变,仅在有必要时进行元素的后移操作。
优缺点:

优点:直接插入排序算法实现简单,适用于小规模数据或部分有序的数据集。对于部分有序的序列,直接插入排序的效率较高,比如对近乎有序的数据进行排序会非常迅速。
缺点:直接插入排序的时间复杂度较高,特别是在处理大规模数据时。对于逆序或近似逆序的数据,算法的性能退化明显,排序时间会很长。
需要注意的是,直接插入排序可以进行原地排序,即在原始数组上进行排序,不需要额外的存储空间。这是因为算法只涉及元素的比较和后移操作,不需要创建新的数组或链表。

代码示例

public class InsertionSort {
    public static void insertionSort(int[] arr) {
        int n = arr.length;
        for (int i = 1; i < n; ++i) {
            int key = arr[i];
            int j = i - 1;

            // 将比当前元素大的元素后移
            while (j >= 0 && arr[j] > key) {
                arr[j + 1] = arr[j];
                j = j - 1;
            }

            arr[j + 1] = key;
        }
    }

    public static void main(String[] args) {
        int[] arr = {5, 2, 8, 12, 3};
        System.out.println("原始数组: ");
        for (int num : arr) {
            System.out.print(num + " ");
        }

        insertionSort(arr);

        System.out.println("\n排序后的数组: ");
        for (int num : arr) {
            System.out.print(num + " ");
        }
    }
}

以上代码定义了一个名为InsertionSort的类,其中包含了一个名为insertionSort的静态方法用于执行直接插入排序。在main方法中,我们创建了一个整型数组arr作为示例输入,并调用insertionSort方法对其进行排序。最终,我们输出排序后的结果。

这段代码中,通过遍历待排序数组,从第二个元素开始逐个将元素插入到已经排好序的子数组中的合适位置,以实现直接插入排序。在插入过程中,对于比当前元素大的元素,我们将其后移一位,直到找到合适的插入位置,然后将当前元素插入该位置。最终,整个数组将按升序排序。

总结

直接插入排序(Insertion Sort)是一种简单直观的排序算法,它的基本思想是将待排序的序列分为已排序和未排序两部分,每次从未排序部分选择一个元素,插入到已排序部分的合适位置,直到所有元素都被插入完毕。以下是直接插入排序的几个要点总结:

实现思路:将待排序序列分为已排序部分和未排序部分。初始时将第一个元素视为已排序部分,剩下的元素视为未排序部分。然后逐个从未排序部分选择元素,通过比较找到插入位置并将其插入到已排序部分,依次重复这个过程直至整个序列有序。

时间复杂度:直接插入排序的最好情况时间复杂度为O(n),最坏情况和平均情况时间复杂度都为O(n^2)。在最好情况下,序列已经有序,只需要进行一次比较即可确定插入位置,所以时间复杂度最低。而在最坏情况下,序列逆序,每个元素都需要与有序序列中的所有元素比较,并可能进行大量的后移操作,导致时间复杂度最高。

空间复杂度:直接插入排序的空间复杂度为O(1),即算法所需的额外存储空间是固定的,不随待排序序列规模的增长而增加。只需要使用常数级别的额外空间来保存一些临时变量。

稳定性:直接插入排序是一种稳定的排序算法。当两个相等的元素进行比较时,如果当前元素小于等于前一个元素,则它们的相对顺序不会改变,仅在有必要时进行元素的后移操作。

优缺点:直接插入排序的优点是实现简单,适用于小规模数据或部分有序的数据集。对于部分有序的序列,直接插入排序的效率较高。然而,直接插入排序的缺点是时间复杂度较高,特别是在处理大规模数据时。对于逆序或近似逆序的数据,算法的性能退化明显,排序时间会很长。

总之,直接插入排序是一种简单但效率略低的排序算法,适用于小规模或部分有序的数据集。它的实现思路清晰,容易理解和实现,同时具备稳定性。然而,在处理大规模数据时,更高效的排序算法如快速排序、归并排序等往往更合适。

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【python基础】—内置模块os常用功能介绍

文章目录 前言一、模块变量os.nameos.environ 二、文件与文件夹os.getcwd(path)os.chdir(path)os.listdir(path)os.mkdir(path)os.remove(path)os.rename(src,dst) 三、os的子模块&#xff1a;Path模块os.path.abspath(path)os.path.basename(path)os.path.dirname(path)os.pat…

软件设计模式——桥接模式

摘要 桥接模式(Bridge pattern): 使用桥接模式通过将实现和抽象放在两个不同的类层次中而使它们可以独立改变。 一、桥接模式的意图 将抽象与实现分离开来&#xff0c;使它们可以独立变化。 二、桥接模式的类图 Abstraction: 定义抽象类的接口Implementor: 定义实现类接口 …

C++标准模板(STL)- 输入/输出操纵符-(std::resetiosflags,std::setiosflags)

操纵符是令代码能以 operator<< 或 operator>> 控制输入/输出流的帮助函数。 不以参数调用的操纵符&#xff08;例如 std::cout << std::boolalpha; 或 std::cin >> std::hex; &#xff09;实现为接受到流的引用为其唯一参数的函数。 basic_ostream::…

【计算机视觉】3.传统计算机视觉方法

传统计算机视觉方法 一、大纲图像分割人脸检测行人检测 二、图像分割基于阈值检测的方法基于边缘检测的方法基于区域的分割方法基于图论的分割方法 三、人脸检测四、行人检测五、SVM六、DPM 一、大纲 图像分割 基于阈值、基于边缘 基于区域、基于图论 人脸检测 Haar-like 特征…

渗透测试之——信息收集思路

请遵守网络安全法 渗透测试时要和客户定好时间再使用扫描器。 渗透测试切记不要渗透客户所给域名的同级域名&#xff0c;可以渗透所给域名的子域名。信息收集永远是渗透测试的第一步。 1. 测试目标 xiusafe.com 2. 域名与ip 渗透测试中IP比域名更适合做渗透&#xff1b; …

Elasticsearch 集群时的内部结构是怎样的?

Apache Lucene : Flush, Commit Elasticsearch 是一个基于 Apache Lucene 构建的搜索引擎。 它利用 Lucene 的倒排索引、查询处理和返回搜索结果等功能来执行搜索。 它还扩展了 Lucene 的功能&#xff0c;添加分布式处理功能以支持大型数据集的搜索。 让我们看一下 Apache Luc…

网络通信(套接字通信)(C/C++)

1.网络编程必知概念 1.广域网和局域网 广域网:又称外网、公网。是连接不同地区局域网或城域网进行计算机通信的远程公共网络。 局域网:在一定的通信范围内,有很个多计算机组成的私有网络就叫局域网。(这些计算机相互之间是可以通信的,但是不能直接访问外网(可以通过网线…

ElementUI -- Mock.js介绍和使用与首页导航栏左侧菜单搭建

1.1 mockjs介绍 Mock.js是一个用于生成随机数据和模拟接口请求的JavaScript库。它可以帮助开发人员在前端开发过程中模拟后端接口的返回数据&#xff0c;以便进行前端页面的开发和测试。 Mock.js有两个重要的特性风靡前端: 数据类型丰富 Mock.js提供了一套简单易用的API&#x…

FL Studio21编曲软件好不好用?值得下载吗

现在的编曲界&#xff0c;鱼龙混杂&#xff0c;很多垃圾软件都来滥竽充数&#xff0c;能入流的编曲软件真的是屈指可数。而编曲业的缺口却又如此之大&#xff0c;所以各类货色都想入行分一杯羹。然而没有实力注定就没有市场而FL Studio又是一款非常著名&#xff0c;实力超群的编…

CTF_BUUCTF_MIS解题_05大白

题目地址&#xff1a;BUUCTF在线评测 文件下载解压之后发现真是一个大白 但是这个大白越看越不对劲&#xff0c;怎么下半身给整没了呢&#xff1f; 随机考虑到图片高度的隐写&#xff0c;杀手锏winhex上场&#xff0c;高度改高一下看看 flag已出现&#xff1a;flag{He1l0_d4_…

面试打底稿③ 专业技能的第三部分

简历原文 抽查部分 基本了解RabbitMQ、Elasticsearch等微服务技术&#xff0c;对分布式事务、分布式缓存等有所学习&#xff1b;熟悉数据库的基本操作&#xff0c;写过C#与数据库交互的项目&#xff1b; 模拟问答 1.你是如何保障消息可靠性的 从消息生产者到交换机&#x…

C语言 数据类型

变量声明 格式&#xff08;变量类型变量名称&#xff09; 变量类型&#xff1a;整数类型&#xff08;int&#xff09;&#xff0c;浮点数类型&#xff08;float&#xff09; float类型可以存储带小数的数字。 用printf()打印变量&#xff0c;使用%d来处理整数值&#xff0c…

AI指令百科全书:1000条AI指令,一次性全给你!

这是一位&#xff0c;国外博主哈桑 整理的&#xff0c;1000条ChatGPT实用指令&#xff0c;涵盖目前几乎所有的&#xff0c;主流提示需求。 全文超过40000字。 我把它们翻译成更适合大家理解的「中文版Prompt」&#xff0c;并根据具体的内容&#xff0c;拆解成一二级目录&…

【C++STL基础入门】list交换、翻转,排序、合并和拼接操作

文章目录 前言一、交换list二、翻转list三、排序list四、合并list五、拼接list总结 前言 在C的标准模板库&#xff08;STL&#xff09;中&#xff0c;list是一个双向链表容器&#xff0c;提供了丰富的功能和操作。本文将介绍list容器在交换、翻转、排序、合并和拼接等方面的基…

Sip多按键对讲分机,洁净室专用对讲终端

Sip多按键对讲分机&#xff0c;洁净室专用对讲终端 嵌入式洁净室电话机广泛应用于手术室&#xff0c;实验室&#xff0c;制药厂车间&#xff0c;无尘车间等 环境要求高的场所&#xff0c;整机是SUS304不锈钢工艺&#xff0c;喇叭&#xff0c;按键&#xff0c;麦克风无间隙&…

三维模型3DTile格式轻量化顶点压缩主要技术方法分析

三维模型3DTile格式轻量化顶点压缩主要技术方法分析 三维模型顶点压缩是3DTile格式轻量化压缩的重要组成部分&#xff0c;能有效减小数据大小&#xff0c;提高数据处理效率。下面将详细分析几种主要的顶点压缩技术方法&#xff1a; 预测性编码&#xff1a;预测性编码也被称为差…

串口电平信号分析--一下看懂不同的串口通信信号

串口电平信号分析–一下看懂不同的串口通信信号

C#,数值计算——Primpolytest的计算方法与源程序

1 文本格式 using System; namespace Legalsoft.Truffer { public class Primpolytest { private int N { get; set; } 32; private int nfactors { get; set; } 5; private ulong[] factors { get; set; } private int[] t { get…

九日集训 LCR.190 加密运算

计算机安全专家正在开发一款高度安全的加密通信软件&#xff0c;需要在进行数据传输时对数据进行加密和解密操作。假定 dataA 和 dataB 分别为随机抽样的两次通信的数据量&#xff1a; 正数为发送量负数为接受量0 为数据遗失 请不使用四则运算符的情况下实现一个函数计算两次…

MySQL索引看这篇就够了

能简单说一下索引的分类吗&#xff1f; 例如从基本使用使用的角度来讲&#xff1a; 主键索引: InnoDB 主键是默认的索引&#xff0c;数据列不允许重复&#xff0c;不允许为 NULL&#xff0c;一个表只能有一个主键。唯一索引: 数据列不允许重复&#xff0c;允许为 NULL 值&…