排序算法:插入排序和希尔排序

一、插入排序

1.基本原理

插入排序(英语:Insertion Sort)是一种简单直观的排序算法。它的工作原理是通过构建有序序列,对于未排序数据,在已排序序列中从后向前扫描,找到相应位置并插入。插入排序在实现上,在从后向前扫描过程中,需要反复把已排序元素逐步向后挪位,为最新元素提供插入空间。

2.动态图

在这里插入图片描述
在这里插入图片描述

3.代码

①:交换方式

public class ThreadNew{
    public static void main(String[] args) {
        int[] arr = new int[] {6,5,3 ,1,8,7,2,4};
        Sort(arr);
    }
    public static void Sort(int[] arr) {
        for (int i = 1; i < arr.length; i++) {
            //将当前数据插入到已经有序的数字当中(这里需要倒着往前找)
            for ( int j = i-1; j >= 0; j--) {
                //前后位置进行交换
                if (arr[j] > arr[j+1]) {
                    int temp = arr[j];
                    arr[j] = arr[j+1];
                    arr[j+1] = temp;
                }else {
                    break;
                }
            }
            System.out.println(Arrays.toString(arr));
        }
    }
}

②:移位方式

public class InsertSort {
    public static void main(String[] args) {
        int[] arr = {-1,100,4,23,2,45,67,89,-3,56};
        System.out.print("排序前:");
        System.out.println(Arrays.toString(arr));
        insertSort(arr);
    }

    //直接插入排序(移位方式)
    public static void insertSort(int[] arr){
        for(int i=1;i<arr.length;i++){
           int insertVal = arr[i];
           int insertIndex = i;
           while (insertIndex>0 && insertVal<arr[insertIndex-1]){
               arr[insertIndex]=arr[insertIndex-1];
               insertIndex--;
           }
           arr[insertIndex] = insertVal;
           //输出每趟的结果
           System.out.print("第" + i + "轮:");
           System.out.println(Arrays.toString(arr));
        }
        //排序完成后输出最终的结果
        System.out.println("==================================================");
        System.out.println(Arrays.toString(arr));
    }
}

二、希尔排序

1.简单插入排序存在的问题

数组arr = {2,3,4,5,6,1}这时要插入的数据1(最小),过程是这样的:
{2,3,4,5,6,6}
{2,3,4,5,5,6}
{2,3,4,4,5,6}
{2,3,3,4,5,6}
{2,2,3,4,5,6}
{1,2,3,4,5,6}
结论:当需要插入得数是较小的数时,后移的次数明显增多,对效率有影响

2.希尔排序介绍

希尔排序也是一种插入排序。它是简单插入排序进过改进之后的一个更高效的版本,也成为了缩小增量排序

3.希尔排序的基本思想

希尔排序是把记录按下标的一定增量分组,对每组使用直接插入排序算法排序;随着增量主键递减,每组包含的关键词也来越多,当增量减至1时,整个文件恰被分成一组,算法终止。如下图:
在这里插入图片描述
在这里插入图片描述

4.交换式 算法实现

public class ShellSort {
    public static void main(String[] args) {
        int arr[] = new int[]{8,9,1,7,2,3,5,4,6,0};
        shellSort(arr);
    }
    // 使用逐步推导的方式来编写希尔排序
    public static void shellSort(int[] arr){
        // 第一轮:是将10个数据分成了5组,这里的 i 值得是我们的步长,
       // 既 我们的 i 指向一组元素当中的 第二个 :注意是第二个
        for (int i = 5 ;i < arr.length;i++){
            // 定义 指针 j : 指向数组当中的每一个元素  默认 j 指向的是该组当中的第一个元素
            // j -= 5 :这里的 5 值得是步长
            for(int j = i-5; j>=0; j -= 5){
                // 如果当前元素大于加上步长之后的元素,则交换
                if(arr[j] > arr[j+5]){
                    int temp = arr[j];
                     arr[j] = arr[j+5];
                     arr[j+5] = temp;
                }
            }
        }


        // 第二轮:是将10个数据分成了2组,这里的 i 值得是我们的步长,
       // 既 我们的 i 指向一组元素当中的 第二个 :注意是第二个
        for (int i = 2 ;i < arr.length;i++){
            // 定义 指针 j : 指向数组当中的每一个元素  默认 j 指向的是该组当中的第一个元素
            // j -= 2 :这里的 2 值得是步长
            for(int j = i-2; j>=0; j -= 2){
                // 如果当前元素大于加上步长之后的元素,则交换
                if(arr[j] > arr[j+2]){
                    int temp = arr[j];
                    arr[j] = arr[j+2];
                    arr[j+2] = temp;
                }
            }
        }

        // 第三轮:是将10个数据分成了1组,这里的 i 值得是我们的步长,
        // 既 我们的 i 指向一组元素当中的 第二个 :注意是第二个
        for (int i = 1 ;i < arr.length;i++){
            // 定义 指针 j : 指向数组当中的每一个元素  默认 j 指向的是该组当中的第一个元素
            // j -= 2 :这里的 2 值得是步长
            for(int j = i-1; j>=0; j -= 1){
                // 如果当前元素大于加上步长之后的元素,则交换
                if(arr[j] > arr[j+1]){
                    int temp = arr[j];
                    arr[j] = arr[j+1];
                    arr[j+1] = temp;
                }
            }
        }
        System.out.println(Arrays.toString(arr));
    }
    
    //最后的整合代码
            for(int gap = arr.length /2;gap > 0;gap /= 2){
                for (int i = gap ;i < arr.length;i++){
                    // 定义 指针 j : 指向数组当中的每一个元素  默认 j 指向的是该组当中的第一个元素
                    // j -= gap :这里的 gap 值得是步长
                    for(int j = i-gap; j>=0; j -= gap){
                        // 如果当前元素大于加上步长之后的元素,则交换
                        if(arr[j] > arr[j+gap]){
                            int temp = arr[j];
                            arr[j] = arr[j+gap];
                            arr[j+gap] = temp;
                        }
                    }
                }
            }

}

完成以后我们来进行一下测试,让他和我们的直接插入排序进行对比。

public static void main(String[] args) {
    int[] arr = new int[80000];
    for(int i = 0;i< 80000;i++){
        arr[i] = (int) (Math.random()*80000);
    }
    
    long startTime=System.nanoTime();   //获取开始时间
    shellSort(arr);
    long endTime=System.nanoTime(); //获取结束时间
    System.out.println("希尔排序运行时间: "+(endTime-startTime)+"ns");

    long startTime1=System.nanoTime();   //获取开始时间
    insertsort(arr);
    long endTime1=System.nanoTime(); //获取结束时间
    System.out.println("直接插入运行时间: "+(endTime1-startTime1)+"ns");
}

在这里插入图片描述
这个时间差距好像不是我们想要看到的结果。
耗时的原因:交换所产生的的耗时
在这里插入图片描述

5.移位式 算法实现

// 这个算法需要仔细去推敲
public static void shellSort2(int[] arr){
    for(int gap = arr.length /2;gap > 0;gap /= 2){
        for (int i = gap ;i < arr.length;i++){
          int j = i;
          int temp = arr[j];
          if(arr[j] <arr[j-gap]){
              while (j - gap >=0 && temp <arr[j-gap]){
                  //移动
                  arr[j] = arr[j-gap];
                  j -= gap;
              }
              arr[j] = temp;
          }
        }
    }
}

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

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

相关文章

27.基于springboot + vue实现的前后端分离-网上租赁交易系统(项目 + 论文)

项目介绍 本课题是根据用户的需要以及网络的优势建立的一个基于Spring Boot的网上租贸系统&#xff0c;来满足用户网络商品租赁的需求。本网上租贸系统应用Java技术&#xff0c;MYSQL数据库存储数据&#xff0c;基于Spring Boot框架开发。在网站的整个开发过程中&#xff0c;首…

学c++对Python有帮助吗?

学习C对Python编程确实有帮助&#xff0c;尽管这两种语言在许多方面有很大的不同。以下是学习C可能对Python编程产生帮助的几个方面&#xff1a; 理解底层概念&#xff1a;C是一种更接近硬件的编程语言&#xff0c;它要求程序员更深入地理解内存管理、指针、数据类型等底层概念…

linux上安装fastdfs及配置

一、基础环境准备 1、所需软件 名称说明libfastcommonfastdfs分离出的一些公用函数包fastdfsfastdas软件包fastdfs-nginx-modulefastdfst和nginx的关联模块nginxnginxl软件包 2、编辑环境 安装一些基础的支持环境 yum install git gccc gcc-c make automake autoconf libto…

个人项目介绍4:三维园区篇

个人项目介绍: 地图铁路线路篇 地球卫星篇 火车站篇 三维园区篇 项目需求&#xff1a; 1.按比例全景显示三维园区 2.精确显示园区内设备设施 3.实时显示设备报警信息 4.显示园区内摄像监控设备&#xff0c;并可点击显示监控视频流 5.显示园区内的重大危险源和风险分布 …

win 11修改通知显示时间

win11toast&#xff1a;python桌面通知工具-CSDN博客

力扣热题100_普通数组_73_矩阵置零

文章目录 题目链接解题思路解题代码 题目链接 73.矩阵置零 给定一个 m x n 的矩阵&#xff0c;如果一个元素为 0 &#xff0c;则将其所在行和列的所有元素都设为 0 。请使用 原地 算法。 示例 1&#xff1a; 输入&#xff1a;matrix [[1,1,1],[1,0,1],[1,1,1]] 输出&…

【金三银四的季节看下Java ORM的走向和性能对比】总结

写在最后 经过将近一周时间的框架收集、学习、实验、编码、测试市面上常见的ORM框架&#xff0c;过程中拜读了很多作者的博文、样例&#xff0c;学习很多收获很多。 重新梳理下整理的框架&#xff1a;mybatis-plus、lazy、sqltoy、mybatis-flex、easy-query、mybatis-mp、jpa、…

Kotlin/Java重写equals后==表现(2)

Kotlin/Java重写equals后表现&#xff08;2&#xff09; 如果不重写默认的equals方法&#xff0c;即使用Object默认的equals()方法&#xff0c;而Object默认的equals方法&#xff0c;其实比较两个对象的地址&#xff1a; fun main(args: Array<String>) {val u1 User(&…

手机app制作商用系统软件开发

手机端的用户占比已经超过了电脑端的用户量&#xff0c;企业想要发展手机端的业务就必须拥有自己的app软件&#xff0c;我们公司就是专门为企业开发手机软件的公司&#xff0c;依据我们多年的开发经验为大家提供在开发app软件的时候怎么选择开发软件的公司。 手机app制…

Java算法之动态规划

Java算法之动态规划 前言 ​ 最近这一段时间一直在刷算法题&#xff0c;基本上一有时间就会做一两道&#xff0c;这两天做了几道动态规划的问题&#xff0c;动态规划之前一直是我比较头疼的一个问题&#xff0c;感觉好复杂&#xff0c;一遇到这样的问题就想跳过&#xff0c;昨…

算法-腐烂的橘子

1、题目来源 994. 腐烂的橘子 - 力扣&#xff08;LeetCode&#xff09; 2、题目描述 在给定的 m x n 网格 grid 中&#xff0c;每个单元格可以有以下三个值之一&#xff1a; 值 0 代表空单元格&#xff1b;值 1 代表新鲜橘子&#xff1b;值 2 代表腐烂的橘子。 每分钟&…

图像处理与视觉感知---期末复习重点(2)

文章目录 一、空间域图像增强1.1 图像增强1.2 几种变换 二、直方图2.1 直方图定义2.2 直方图均衡化2.3 离散情况2.4 例子2.5 直方图匹配2.6 例子2.7 一道例题 三、空间滤波器3.1 定义3.2 例子 四、平滑空间滤波器4.1 作用与分类4.2 线性滤波器 五、统计排序滤波器5.1 定义与分类…

CleanMyMac X4.14.7永久免费Mac电脑清理和优化软件

CleanMyMac X 是一款功能强大的 Mac 清理和优化软件&#xff0c;适合以下几类人群使用&#xff1a; 需要定期清理和优化 Mac 的用户&#xff1a;随着时间的推移&#xff0c;Mac 设备上可能会积累大量的无用文件、缓存和垃圾&#xff0c;导致系统运行缓慢。CleanMyMac X 的智能扫…

谷歌开源的LLM大模型 Gemma 简介

相关链接&#xff1a; Hugging face模型下载地址&#xff1a;https://huggingface.co/google/gemma-7bGithub地址&#xff1a;https://github.com/google/gemma_pytorch论文地址&#xff1a;https://storage.googleapis.com/deepmind-media/gemma/gemma-report.pdf官方博客&…

傅里叶变换pytorch使用

参考视频&#xff1a;1 傅里叶变换原理_哔哩哔哩_bilibili 傅里叶得到低频、高频信息&#xff0c;针对低频、高频处理能够实现不同的目的。 傅里叶过程是可逆的&#xff0c;图像经过傅里叶变换、逆傅里叶变换后&#xff0c;能够恢复到原始图像 在频域对图像进行处理&#xff0c…

如何建立一个电商网站呢?商城完善建站解决方案

如何建立一个电子商务网站&#xff1f; 电子商务网站是虚拟商店&#xff0c;用户可以不受时间和地点的限制在线购物。 网上商城的兴起&#xff0c;扩大了消费市场空间。 一个完善的网站建设解决方案对于商城来说是必不可少的。 以下是为您提供的一些网站建设建议&#xff1a; …

[uni-app ] createAnimation锚点旋转 及 二次失效问题处理

记录一下: 锚点定位到左下角, 旋转动画 必须沿Z轴,转动 但是,此时会出现 后续动画在微信小程序失效问题 解决: 清空 this.animationData

LiveNVR监控流媒体Onvif/RTSP功能-支持云端录像监控视频集中存储录像回看录像计划配置NVR硬件设备录像回看

LiveNVR支持云端录像监控视频集中存储录像回看录像计划配置NVR硬件设备录像回看 1、流媒体服务软件2、录像回看3、查看录像3.1、时间轴视图3.2、列表视图 4、如何分享时间轴录像回看&#xff1f;5、iframe集成示例7、录像计划7、相关问题7.1、录像存储位置如何配置&#xff1f;…

Spring MVC 全局异常处理器

如果不加以异常处理&#xff0c;错误信息肯定会抛在浏览器页面上&#xff0c;这样很不友好&#xff0c;所以必须进行异常处理。 1.异常处理思路 系统的dao、service、controller出现都通过throws Exception向上抛出&#xff0c;最后由springmvc前端控制器交由异常处理器进行异…

Django高级之-缓存

Django高级之-缓存 一 缓存介绍 在动态网站中,用户所有的请求,服务器都会去数据库中进行相应的增,删,查,改,渲染模板,执行业务逻辑,最后生成用户看到的页面. 当一个网站的用户访问量很大的时候,每一次的的后台操作,都会消耗很多的服务端资源,所以必须使用缓存来减轻后端服务…