【JAVA入门】Day24 - 排序算法

news2025/1/18 11:43:49

【JAVA入门】Day24 - 排序算法


文章目录

  • 【JAVA入门】Day24 - 排序算法
    • 一、冒泡排序
    • 二、选择排序
    • 三、插入排序
    • 四、快速排序
      • 4.1 递归
      • 4.2 快速排序


        排序,是把混乱的数据排成从小到大或从大到小。
        排序一共有十种左右,它们是:冒泡排序、选择排序、插入排序、快速排序、希尔排序、归并排序、堆排序、计数排序、桶排序、基数排序
        在这里,我们着重介绍前四种。

一、冒泡排序

        冒泡排序:相邻的数据两两比较,小的放前面,大的放后面。
        每经过一轮的两两比较,最大的数据一定会被放在最右边。此时只需要对除它之外的剩余元素继续做相同操作即可。第二轮可以比第一轮少循环一次,以此类推。
        如果数组中有 n 个数据,我们总共只需要执行 n-1 轮的代码就可以了。

package SortTest;

public class SortDemo1 {
    public static void main(String[] args) {
        //BubbleDemo

        //1.定义数组
        int[] arr = {2, 4, 5, 3, 1};

        //2.利用冒泡排序将数组中的数据变成 1 2 3 4 5
        //注意循环的截止要减一,防止越界
        for(int i = 0; i < arr.length - 1; i++) {
            for (int j = 0; j < arr.length - 1 - i; j++) {
                if (arr[j] > arr[j + 1]) {
                    int temp = arr[j];
                    arr[j] = arr[j + 1];
                    arr[j + 1] = temp;
                }
            }
        }
        for (int i = 0; i < arr.length; i++) {
            System.out.print(arr[i] + " ");
        }

    }
}

二、选择排序

        选择排序:从0索引开始,拿着每一个索引上的元素,跟后面的元素依次比较,小的放前面,大的放后面,依此类推。
        每一轮循环结束后,最小的数据一定是跑到了左边,那么后面的下一轮循环就可以少循环一次。

package SortTest;

public class SortDemo2 {
    public static void main(String[] args) {
        //SelectionSort

        //1.定义数组
        int[] arr = {2, 4, 5, 3, 1};

        //2.利用选择排序让数组变成正序
        //从0索引开始,跟后面的元素一一比较
        for(int i = 0; i < arr.length - 1; i++) {
            for (int j = i + 1; j < arr.length; j++) {
                if (arr[i] > arr[j]) {
                    int temp = arr[i];
                    arr[i] = arr[j];
                    arr[j] = temp;
                }
            }
        }
        for (int i = 0; i < arr.length; i++) {
            System.out.print(arr[i] + " ");
        }
    }
}

三、插入排序

        插入排序:我们将0索引的元素到N索引的元素看作是有序的,把N+1开始的索引的元素到最后一个元素当成是无序的。遍历无序的元素,我们将遍历到的元素插入到有序序列中适当的位置,若遇到相同数据了,插在后面。N的取值范围:0~最大索引。
        每次插入时,我们是从后往前插,分别跟当前索引的数据比大小,比它小,往前继续走,直到找到比当前索引大的情况,插在后面。

package SortTest;

public class SortDemo3 {
    public static void main(String[] args) {
        //插入排序:我们将0索引的元素到N索引的元素看作是有序的,把N+1开始的索引的元素到最后一个元素当成是无序的。遍历无序的元素,我们将遍历到的元素插入到有序序列中适当的位置,若遇到相同数据了,插在后面。N的取值范围:0~最大索引。

        int[] arr = {3, 44, 38, 5, 47, 15, 36, 26, 27, 2, 46, 4, 19, 50, 48};

        //1.找到无序的哪一组数组是从哪个数组开始的
        int startIndex = -1;
        for(int i = 0; i < arr.length - 1; i++) {
            if(arr[i] > arr[i + 1]) {
                //System.out.println(i); //有序的那一组数据到i索引就结束了,再加一就是无序数据的第一个索引
                startIndex = i + 1;
                break;
            }
        }

        //2.遍历从startIndex开始到最后一个元素,得到无序的每一个元素
        for (int i = startIndex; i < arr.length; i++) {
            //问题:如何把遍历到的数据插入到前面有序的这一组当中

            //记录当前要插入数据的索引
            int j = i;

            //将每一个无序数据向左遍历,直到找到比它小的档口,插入
            while(j > 0 && arr[j] < arr[j - 1]) {
                int temp = arr[j];
                arr[j] = arr[j - 1];
                arr[j - 1] = temp;
                j--;
            }
        }

        for (int i = 0; i < arr.length; i++) {
            System.out.print(arr[i] + " ");
        }
    }

}

四、快速排序

4.1 递归

        递归指的是方法中调用方法本身的现象。
        递归一定要有出口,否则就会出现内存溢出。
        递归算法可以把一个复杂的问题层层转化为一个与原问题相似的规模较小的问题来求解。
【练习1】利用递归求1~100之间的和。

package SortTest;

public class RecursionDemo {
    public static void main(String[] args) {
        //求1~100的和
        //1~100的和 = 100 + (1~99的和)
        //1~99的和 = 99 + (1~98的和)
        //...
        //1~2的和 = 2 + (1~1的和)
        //1~1的和 = 1 (递归的出口)
        System.out.println(getSum(100));
    }
    public static int getSum(int number){
        if(number == 1) {
            return 1;
        }
        return number + getSum(number - 1);
    }

}

【练习2】用递归求5!。
5! = 5 * 4 * 3 * 2 * 1

package SortTest;

public class RecursionDemo1 {
    public static void main(String[] args) {
        //5! = 5 * 4!
        //4! = 4 * 3!
        //3! = 3 * 2!
        //2! = 2 * 1!
        //1! = 1

        System.out.println(getFactorialRecursion(5));

    }

    public static int getFactorialRecursion(int number) {
        if(number == 1) {
            return 1;
        }

        return number * getFactorialRecursion(number - 1);
    }
}

4.2 快速排序

        第一轮:把0索引的数字作为基准数,确定基准数在数组中正确的位置。比基准数小的全部在左边,比基准数大的全部在右边。
在这里插入图片描述
在这里插入图片描述
        找到 start 和 end 的位置后,交换两个位置上的数据。
在这里插入图片描述
        直到 start 和 end 指向同一个位置时,这个位置就是基准数要存入的位置(基准数归位)。
在这里插入图片描述
        此时就需要把基准数和它们指向的位置交换。
在这里插入图片描述
        此时,6 左侧的数据都比 6 小,6 右侧的数据都比 6 大。

package SortTest;

public class SortDemo4 {
    public static void main(String[] args) {
        //QuickSort
        int[] arr = {6, 1, 2, 7, 9, 3, 4, 5, 10, 8};

        quickSort(arr, 0, arr.length - 1);

        for (int i = 0; i < arr.length; i++) {
            System.out.print(arr[i] + " ");
        }

    }

    /*
    参数一:要排序数组的起始数组
    参数二:要排序数组的起始索引
    参数三:要排序数组的结束索引
     */
    public static void quickSort(int[] arr, int i , int j) {
        //定义两个变量记录要查找的范围
        int start = i;
        int end = j;

        //递归出口
        //start > end
        if(start > end) {
            return;
        }

        //记录基准数
        int baseNumber = arr[i];

        //利用循环找到要交换的数字
        while(start != end) {
            //利用end,从后往前开始找,找比基准数小的数字
            while(true) {
                if(end <= start || arr[end] < baseNumber) {
                    break;
                }
                end--;
            }

            //利用start,从前往后找,找比基准数大的数字
            while(true) {
                if(end <= start || arr[start] > baseNumber) {
                    break;
                }
                start++;
            }

            //把end和start指向的元素进行交换
            int temp = arr[start];
            arr[start] = arr[end];
            arr[end] = temp;

            //直到start == end,循环结束
        }
        //表示已经找到了基准数在数组中应该存入的位置
        //基准数归位
        int temp = arr[i];
        arr[i] = arr[start];
        arr[start] = temp;

        //确定基准数左边的范围,重复刚刚所有的事情
        quickSort(arr, i, start - 1);
        //确定基准数右边的范围,重复刚刚所有的事情
        quickSort(arr, end + 1, j);

    }
}

        在所有排序算法中,快速排序的速度是最快的,被广泛使用。

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

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

相关文章

Salesforce 发布开源大模型 xGen-MM

xGen-MM 论文 在当今 AI 技术飞速发展的时代&#xff0c;一个新的多模态 AI 模型悄然崛起&#xff0c;引起了业界的广泛关注。这个由 Salesforce 推出的开源模型—— xGen-MM&#xff0c;正以其惊人的全能特性和独特优势&#xff0c;在 AI 领域掀起一阵旋风。那么&#xff0c;x…

书生大模型(第3期)基础岛第5关--XTuner 微调个人小助手认知

XTuner微调前置基础 1 基本概念 在进行微调之前&#xff0c;我们需要了解一些基本概念。 1.1 Finetune简介 微调&#xff08;fine-tuning&#xff09;是一种基于预训练模型&#xff0c;通过少量的调整&#xff08;fine-tune&#xff09;来适应新的任务或数据的方法。 微调…

VUE3 无法修改 el-dialog 样式

用下面这种方式修改 el-dialog 组件样式一点作用都没有&#xff0c;正常用这种方式修改 el 的el-button、tab等都是百试不爽的。最后找到解决办法和原因。在el-dialog外面套一层div /deep/ .el-dialog { background: url(https://lanhu-oss.lanhuapp.com/7cbd761cd26f7b255086…

Cesium for Unreal——第四节 Transition Between Locations on the Globe 在两个位置间平稳飞行

文章目录 1. 创建或打开上次的项目2. 蓝图3. 构建一个新的关卡 Globel Level,添加墨尔本地形数据4. 选择蓝图文章参考与Cesium官网 Transition Between Locations on the Globe 1. 创建或打开上次的项目 学习之前,需要先安装——创建项目——运行 UE ,点击跳转 2. 蓝图 蓝图…

AI工具集合

AI工具集官网 | 1000 AI工具集合&#xff0c;国内外AI工具集导航大全

QT下显示自己派生的QWidget界面(提升为)

在实际开发过程中&#xff0c;我们可能有这样的需求&#xff0c;自己绘制一个仪表盘界面&#xff0c;然后将其贴到主界面上方。 这个时候就会用到“提升为”这个功能&#xff0c;该功能目的是将QWidget提升为自己派生的QWdiget子类&#xff0c;具体操作为&#xff0c;在主界面…

元数据管理gravitino学习

元数据管理的组成有几个部分&#xff1a;Metaservice(Gravitino)、Luoshu&#xff08;amoro)、Hive Metastore&#xff0c;其中gravitino是数据管理模块实现元数据统一管理的核心。前面有提到hive metastore可以存储hive的库表元数据信息&#xff0c;可以用于存储关于hive表、列…

19 自定义类型:结构体、联合体、枚举

目录 一、结构体 &#xff08;一&#xff09;结构体类型的定义 &#xff08;二&#xff09;结构体变量的创建和初始化 1、结构体变量的创建 &#xff08;1&#xff09;定义完结构体后再创建变量 &#xff08;2&#xff09;在定义结构体的同时创建变量 &#xff08;3&…

代码随想录算法day19 | 回溯算法part01 | 77. 组合,216.组合总和III,17.电话号码的字母组合

第77题. 组合 对着 在 回溯算法理论基础 给出的 代码模板&#xff0c;来做本题组合问题&#xff0c;大家就会发现 写回溯算法套路。 力扣题目链接(opens new window) 给定两个整数 n 和 k&#xff0c;返回 1 ... n 中所有可能的 k 个数的组合。 示例: 输入: n 4, k 2 输出: […

使用Xshell6远程登录Linux 服务器--远程上传下载文件Xftp6的使用

&#x1f600;前言 本篇博文是关于Linux 实操篇-使用Xshell6远程登录Linux 服务器–远程上传下载文件Xftp6的使用&#xff0c;希望你能够喜欢 &#x1f3e0;个人主页&#xff1a;晨犀主页 &#x1f9d1;个人简介&#xff1a;大家好&#xff0c;我是晨犀&#xff0c;希望我的文章…

“跨越国界,共赢未来:跨境联盟营销的策略与实践

全球化背景下跨境联盟营销的市场有很多机遇&#xff0c;随着全球化的深入发展&#xff0c;跨境电商市场不断扩大&#xff0c;为企业提供了广阔的市场空间。跨境联盟营销可以借助全球化的趋势&#xff0c;实现品牌国际化和市场拓展。随着大数据、人工智能、区块链等技术的不断发…

16款热门WMS 智能仓储管理系统盘点,助力企业数字化转型!

你是否想过&#xff0c;一个企业的仓库就如同其心脏般重要&#xff1f;而 WMS 智能仓储管理系统正是确保这颗 “心脏” 高效跳动的关键。它不仅能精准管理库存&#xff0c;实现货物的快速出入库&#xff0c;还能优化库位分配&#xff0c;提高仓库空间利用率。通过实时的数据监控…

【Linux】冯诺依曼体系|操作系统概念

目录 一、冯诺依曼体系结构 注意事项 存储器的意义&#xff1a;缓冲 数据流动事例 二、操作系统 操作系统的概念 操作系统的定位与目的 操作系统的管理 系统调用和库函数概念 一、冯诺依曼体系结构 冯诺依曼架构&#xff08;von Neumann architecture&#xff09;是一…

算法全面剖析

算法 查找算法&#xff1a; 顺序查找&#xff1a; 基本思想&#xff1a; 顺序查找也称为线形查找&#xff0c;属于无序查找算法。从数据结构线形表的一端开始&#xff0c;顺序扫描&#xff0c;依次将扫描到的结点关键字与给定值k相比较&#xff0c;若相等则表示查找成功&am…

2024年全新基于Java爬取微博数据(完整版)

2024年全新基于Java爬取微博数据(完整版) 爬虫背景爬虫分析爬取微博主页正文列表数据引入jar包编写代码关于微博 Cookie处理文本的正则微博正文长文本补全什么是正文长文本获取正文长文本编写代码导出微博数据到Excel引入jar包编写代码突来的疑问微博正文内容分析转存 图片 o…

继电器介绍及qt操作继电器实战

一.继电器基础介绍 32路继电器通常用于自动化控制系统中&#xff0c;能够同时控制多达32个不同的电气设备。以下是对32路继电器的一些详细介绍&#xff1a; 1. 基本概念 继电器&#xff1a;一种电气控制装置&#xff0c;当输入信号&#xff08;通常是电流或电压&#xff09;…

浏览器请求无缝导入apifox(无需客户端,在线使用)方法

不用下载客户端浏览器在线模拟请求&#xff0c;方便快捷&#xff01; 废话不多少&#xff0c;只需三步&#xff0c;往下看&#xff01; 一步&#xff1a; 打开F12 -> 网络 -> 找到你要模拟的请求&#xff0c;右键以cURL格式复制 二步&#xff1a; Apifox 访问Apifox的w…

“浙里办”统一用户组件-接入 基于单点登录票据换取请求 token

基于单点登录票据换取请求 token 1. 请求地址 政务外网地址: https://bcdsg.zj.gov.cn:8443/restapi/prod/IC33000020220329000007/ uc/sso/access_token 互联网地址: https://ibcdsg.zj.gov.cn:8443/restapi/prod/IC33000020220329000007 /uc/sso/access_token 2. 入参…

AFSim 仿真系统----集成指南

引言 本文档描述了将新功能扩展和集成到 AFSIM 中的有限方法。允许并描述多种方法&#xff0c;以及在 AFSIM 社区标准和指南中引入集成作为可共享资源的要求。 概述 核心可执行文件 基于 AFSIM 的可执行文件通常由单个 AFSIM “应用程序” 组成。该应用程序维护脚本类型、扩…

Linux rocky 9.2 安装mysql-8.0.39-linux-glibc2.28-x86_64.tar.xz

数据库官方下载&#xff1a;MySQL :: Download MySQL Community Server 本文也绑定该资源包&#xff0c;免费提供下载学习。 1.系统版本 2.新建目录&#xff0c;存放数据库安装包&#xff0c;并且上传 需要用到的工具&#xff1a;yum -y install vim lrzsz tar 上传解压&…