排序算法及实现(上)

news2025/1/22 18:08:56
稳定性的判断:如果两个相同大小的元素也进行了交换就是不稳定,否则稳定
1.直接插入排序:

当插入第 i 位置元素时,前面 0 到 i-1 位置的元素已经各自有序。

此时将i 再次从i-1到0位置依次进行比较。找到合适位置将其插入,原本位置上的元素后移。

插入排序性质:

1.元素越接近有序,其效率越高。

2.时间复杂度:O(N^2)

3.空间复杂度:O(1)   //不占用额外的空间

4.稳定性:稳定          //一般判断是否稳定的标准是 遇到相同的元素是否交换,不交换则稳定

代码实现:

public static void insertSort(int[]array){

        //i从第二个位置开始遍历
    for(int i=1;i<array.length;i++){

        //每次外层循环都将i的值记录下来
        int temp = array[i];
        int j = i-1;

        //j从i的后一个元素开始,依次向后遍历
        for(;j>=0;j--){
            if(array[j] > temp){
                array[j+1] = array[j];
            }else{
                
                如果j的值小于或等于i直接跳出,继续i遍历
                break;
            }
        }
        array[j+1] = temp;
    }
}

2.希尔排序(缩小增量版排序)

思想:先选定一个整数,把待排序的数据按照这个整数的大小进行分组

再依次对各个组内的数据进行排序。这个整数不断减小,直到为1时,将所有数据

放在同一组排序。

下图为每次分组排序的过程。当gap=1 时排序完成,依次交换画下线的元素

总结:

1.希尔排序是对插入排序的优化,

2.当gap = 1之前,都是对数组的预排序目的是让数组更加接近有序,

        对整体而言当gap = 1时可以很好的排序,达到优化效果

3.希尔的时间复杂度不容易计算,因为gap的取值有很多。

        大多数的书给出的希尔排序的时间复杂度都不同,一般按照O(n^1.25)到O(6*n^1.25)计算

4.稳定性:不稳定

代码实现:

        

 /**
     * 希尔排序(分组后 用到插入排序)
     * 稳定性:不稳定
     * 时间复杂度:logN
     * @param arry
     */
    public static void shellSort(int[] arry){
        //分组 gap
        int gap = arry.length;
        while(gap > 1){
            gap =gap/3+1;
            shell(arry,gap);
        }
    }

    /**
     * 每组进行插入排序
     * @param array
     * @param gap
     */
    private static void shell(int[] array,int gap){
        for (int i = gap; i < array.length; i++) {
            int temp = array[i];
            int j = i-gap;
            for(;j>=0;j-=gap){
                if(array[j] > temp){
                    //将小的赋值给大的
                    array[j+gap] = array[j];
                }
                else {
                    break;
                }
            }
            //不小就自己给自己赋值
            array[j+gap] = temp;
        }
    }

3.选择排序

思想:每次从排序的元素中选择最小或者最大的,放在序列的起始位置,直到要排序的元素全部拍完。

缺陷:效率不高,实际很少使用

时间复杂度:O(N^2),最坏情况每个元素两两之间都要比较

空间复杂度:O(1)

稳定性:不稳定

代码如下:

     */
    public static void swap(int[]array,int i,int j){
        int temp = array[j];
        array[j]  = array[i];
        array[i] = temp;
    }
    public static void selectSort(int[]array){
        for(int i=0;i< array.length;i++){
            int mindex = i;
            //从第二个位置开始寻找最下值的线标
            for(int j=i+1;j< array.length;j++){
                if(array[j]<array[mindex]){
                    mindex = j;
                }
            }
            //找到的最小值下标与当前位置交换
            swap(array,mindex,i);
        }
    }
}

代码二:


4.堆排序(以大根堆为例)

排升序建立大根堆,排降序建立小根堆

时间复杂度:O(n*logn)  每一次的排序为logn,进行n次排序

空间复杂度:O(1)  //不占用额外的空间

稳定性:不稳定

 /**
     * 升序建立大根堆
     */
    public static void headSort(int[]array){

        bigHead(array);
        int end = array.length-1;
        while (end>=0){
            swap(array,0,end);
            shiftDown(array, end,0 );
            end--;
        }
    }

    /**
     * 建立大根堆
     * @param arr
     */
    public static void bigHead(int[]arr){
        for (int parent = (arr.length-1-1)/2; parent >=0 ; parent--) {
            shiftDown(arr,arr.length-1,parent);
        }
    }

    /**
     *向下排序
     * @param array
     * @param end
     * @param parent
     */
    public static void shiftDown(int[]array,int end,int parent){
        //左子树
        int child = 2*parent+1;
        while (child < end){
            //先判断右节点是否存在
            if(child+1 <end && array[child+1] >array[child]){
                child +=1;
            }
            if(array[child] > array[parent]){
                swap(array,child,parent);
                parent = child;
                child = 2*parent+1;
            }else {
                break;
            }
        }
    }

  /**
     * 用来交换数据的函数
     * @param array
     * @param i
     * @param j
     */
    public static void swap(int[]array,int i,int j){
        int temp = array[j];
        array[j]  = array[i];
        array[i] = temp;
    }

5.冒泡排序

时间复杂度:O(N^2)

空间复杂度:O(1)

稳定性:稳定

代码实现:

public static void bubbleSort(int[]array){
        for (int i = 0; i < array.length-1; i++) {
            //定义flag进行优化,如果一趟下来都没有交换元素,则已经有序
            //直接break跳出,进行下一趟的比较
            boolean flag = false;
            for (int j = 1; j <array.length-i-1 ; j++) {
                if(array[j] > array[j+1]){
                    swap(array,j,j+1);
                    flag = true;
                }
            }
            //在优化的情况下如果数据 1 2 3 4 5
            //时间复杂度为:O(N)
            if(flag == false){
                break;
            }
        }
    }

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

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

相关文章

Java 【数据结构】 哈希(Hash超详解)HashSetHashMap【神装】

登神长阶 第十神装 HashSet 第十一神装 HashMap 目录 &#x1f454;一.哈希 &#x1f9e5;1.概念 &#x1fa73;2.Object类的hashCode()方法: &#x1f45a;3.String类的哈希码: &#x1f460;4.注意事项: &#x1f3b7;二.哈希桶 &#x1fa97;1.哈希桶原理 &#x…

AI地名故事:沧联村

沧联村&#xff0c;位于黄埔区云埔街&#xff0c;与开发区东区、增城区接壤&#xff0c;辖区面积约6.58平方公里。这个村庄的历史悠久&#xff0c;充满了丰富的故事。 在很久以前&#xff0c;沧联村并未有现今的名称。然而&#xff0c;随着时间的流转&#xff0c;村庄逐渐形成…

美股市场恒生指数冲刺19000点关口 地产股大涨

查查配5月10日电(中新财经记者 谢艺观)5月10日,港股现强势行情,恒生指数盘中一度冲至18993.28点,距离19000点关口仅一步之遥。 美港通证券以其专业的服务和较低的管理费用在市场中受到不少关注。该平台提供了实盘交易、止盈止损、仓位控制等功能,旨在为投资者提供更为全面的投…

深度学习设计模式之单例模式

一、单例模式简介 一个类只能有一个实例&#xff0c;提供该实例的全局访问点&#xff1b; 二、单例模式实现步骤 使用一个私有构造函数、一个私有静态变量以及一个公有静态函数来实现。 私有构造函数保证了不能通过构造函数来创建对象实例&#xff0c;只能通过公有静态函数返…

验证码生成--kaptcha

验证码生成与点击重新获取验证码 如图所示&#xff0c;本文档仅展示了验证码的生成和刷新显示。 1. 概述 系统通过生成随机验证码图像和文本。 2. 代码分析 2.1. Maven依赖 <dependency><groupId>com.github.penggle</groupId><artifactId>kaptch…

VirtualBox7安装ubantu server 22.04通过NAT+Only-Host双网卡实现宿主机与虚拟机互通

目录 背景环境安装虚拟机配置网卡修改ssh端口遇到的坑参考文章 背景 时间长没用docker了&#xff0c;有些命令都快忘了&#xff0c;心血来潮想着搞个docker玩一玩&#xff0c;所以需要先搞一个虚拟机&#xff0c;因为之前CentOS用的比较多&#xff0c;所以这次想试一试ubantu。…

Java入门——继承和多态(上)

包 包是组织类的一种方式. 使用包的主要目的是保证类的唯一性. 例如, 你在代码中写了一个 Test 类. 然后你的舍友也可能写一个 Test 类. 如果出现两个同名的类, 就会冲突, 导致 代码不能编译通过. 导入包中的类 Java 中已经提供了很多现成的类供我们使用. 例如 public cla…

【C -> Cpp】由C迈向Cpp (5)

标题&#xff1a;【C -> Cpp】由C迈向Cpp&#xff08;5&#xff09; 水墨不写bug &#xff08;图片来源于网络&#xff09; 不抵制失败&#xff0c;携手失败&#xff0c;迈向成功 正文开始&#xff1a; &#xff08;一&#xff09;深入理解构造函数 在之前的讲解中&#x…

linux系统(ubuntu)调用科大讯飞SDK实现语音识别

1. 科大讯飞官网 登录注册实名制 2. 点击控制台&#xff0c;创建应用 点击左侧的语音听写&#xff0c;右边下滑选择Linux&#xff0c;点击下载 选择Linux平台&#xff0c;普通版本&#xff0c;语音听写&#xff0c;SDK下载 此时将得到一个压缩包&#xff0c;选择的功能不…

LVS的三种工作模式---(DR/TUN/NAT)

目录 一、NAT模式&#xff08;LVS-NAT&#xff09; 二、IP隧道模式&#xff08;LVS-TUN&#xff09; 三、DR模型--直接路由模式&#xff08;LVS-DR&#xff09; LVS/DR模式ARP抑制 原因&#xff1a; LVS的DR工作模式及配置&#xff1a; LVS的NAT工作模式及配置&#xff1…

邂逅Linux--常见指令,万物为文件(一)

引子&#xff1a;在之前&#xff0c;我们经常听到Linux&#xff0c;那什么是Linux呢&#xff1f;Linux是一种免费使用和自由传播的类UNIX操作系统&#xff0c;其内核由林纳斯本纳第克特托瓦兹&#xff08;Linus Benedict Torvalds&#xff09;于1991年10月5日首次发布&#xff…

命令行工具部署达梦数据库 DMDPC(BP 多副本架构)

解达梦数据库DPC集群的主要使用场景&#xff1a; DMDPC 关注和解决的是大数据、计算与存储分离、高可用、支持全部的 SQL 标准、拥有完整的事务处理能力和集群规模能够动态伸缩的业务场景&#xff1a; 大量的复杂查询操作要求优化器能够生成优良的执行计划&#xff0c;并且执…

0基础理解ECC并做题-攻防世界easy-ECC理解

基点p就是最初选定的那个点 1和2都是整数集合&#xff0c;但是1/20.5就不属于整数集合 一直加&#xff0c;一直乘&#xff0c;还能保证有限个数字&#xff1f;这是因为采用了取模的运算&#xff0c;让元素始终都在有限的范围内。 如何计算分数求模&#xff1f; 设n1/2mod23,那么…

使用注解的方式进行配置RabbitMQ

引入依赖&#xff1a; <dependency><groupId>org.springframework.amqp</groupId><artifactId>spring-rabbit-test</artifactId><scope>test</scope></dependency> 配置application.yml server:port: 8082 spring:rabbitmq…

Coze扣子开发指南:AI零代码编程创建插件

在Coze扣子中创建插件&#xff0c;有两种方式&#xff0c;一是用API&#xff0c;具体方式参照上一篇文章《Coze扣子开发指南&#xff1a;用免费API自己创建插件》&#xff0c;还有一种方式就是编程&#xff0c;不过有了AI的帮助&#xff0c;即使不会编程的人&#xff0c;也可以…

如何清除DNS缓存,刷新DNS

大家在使用域名访问服务器的时候&#xff0c;经常会遇到一个问题&#xff0c;同一个局域网里的两台电脑&#xff0c;一台可以访问而另一台不行。这是为什么呢&#xff1f;这里我要和大家说下DNS缓存的问题&#xff0c;顾名思义&#xff0c;每台电脑都有DNS缓存&#xff0c;在域…

MyBatis-plus(一):快速入门

目录 一、MyBatis-plus 快速入门 1、原理 2、实体类命名规则 3、常见注解 4、主键 id 策略 5、使用 TableField 的常见场景 6、常用配置 二、核心功能 1、条件构造器 2、自定义 SQL 3、IService 接口 一、MyBatis-plus 快速入门 1、原理 MyBatisPlus 通过扫描实体…

Leetcode 剑指 Offer II 077.排序链表

题目难度: 中等 原题链接 今天继续更新 Leetcode 的剑指 Offer&#xff08;专项突击版&#xff09;系列, 大家在公众号 算法精选 里回复 剑指offer2 就能看到该系列当前连载的所有文章了, 记得关注哦~ 题目描述 给定链表的头结点 head &#xff0c;请将其按 升序 排列并返回 排…

STM32入门_江协科技_5~6_OB记录的自学笔记_GPIO输出_LED流水灯_蜂鸣器

5. GPIO 输出 5.1. GPIO简介 GPIO&#xff08;General Purpose Input Output&#xff09;通用输入输出口可配置为8种输入输出模式引脚电平&#xff1a;0V~3.3V&#xff0c;部分引脚可容忍5V&#xff08;端口输入5V的电压&#xff0c;之前引脚定义表格中带FT标识的&#xff09…

暗区突围哪里获得测试资格 暗区突围测试资格获取方法

在游戏业界的浩瀚星空中&#xff0c;《暗区突围》如同一颗璀璨新星&#xff0c;以其独树一帜的游戏模式和前所未有的沉浸式体验&#xff0c;迅速吸引了全球玩家的目光。它不仅仅是一款游戏&#xff0c;更像是一次对勇气、智慧与团队合作的深度探索。玩家在危机四伏的暗区中&…