算法第4版 第2章排序

news2024/11/28 2:56:15

综述:5个小节,四种排序+应用,初级排序、归并排序、快速排序、优先队列

===2.1.初级排序===

排序算法模板,less(), exch(), 排序代码在sort()方法中;

选择排序:如升序排列,1.找到数组中最小的元素;2.与数组中的第一个元素交换下位置;3.再在剩下的元素找最小值,与第二个元素交换下位置;重复,直到所有元素都完成排序。

交换的总次数是n,算法的时间效率取决于比较次数(N2/2);

选择排序时间复杂度(O(n2)),两个特点:运行时间与输入无关;数据移动是最少的;

选择排序的实现:

private static boolean less(Comparable v, Comparable w){

return v.compareTo(w)<0;

}

public static void exch(Comparable[]a, int i, int j){

Comparable t = a[i]; a[i]=a[j];a[j]=t;

}

public class Selection{

public static void sort(Comparable[]){

int N = a.length;

for(int i = 0; i<N;i++){

int min =i;

for(int j =i+1;j<N;j++){

if(less a[j], a[min]) min =j;

exch(a, i, min);

}

}

}

冒泡排序:一次比较两个元素,如果位置不对就交换,一次遍历要交换多次

插入排序:第i个元素与第i之前的元素对比,插入到合适的位置。左侧部门一直为有序数组

对于随机排列的长度未N且主键不重复的数组,平均情况下插入排序需要N2/4比较,N2/4次交换。最坏情况下需亚奥N2/2, N2/2次交换。插入排序对有序数组的排序,运行时间是线性的。

代码实现:

publIc class Insertion{

public static void sort(Comparable[] a){

int N = a.length;

for(int i = 1; i<N; i++){

for(int j =1; j>0&&less(a[j], a[j-1]); j--)

exch(a, j, j-1);

}

}

}

部分有序数组:数组中的每个元素距离它的最终位置都不远;一个有序的大数组接一个小数组;数组中只有几个元素的位置不正确。

希尔排序的思想是使数组中任意间隔为h的元素都是有序的。这样的数组被称为h有序数组。换句话说,一个h有序数组就是h个互相独立的有序数组编织在一起组成的一个数组(见图2.1.2)。在进行排序时,如果h很大,我们就能将元素移动到很远的地方,为实现更小的h有序创造方便。用这种方式,对于任意以1结尾的h序列,我们都能够将数组排序。这就是希尔排序。

代码实现:

public class Shell{

public static void sort(Comparable[] a){

int N = a.length;

int h =1;

while(h<N/3) h = 3*h+1;

while(h>=1){

for(int i =h; i<N; i++){

for(int j=i, j>=h&&less(a[j],a[j-h]); j -=h)

exch(a, j, j-h);

}

h = h/3;

}

}

}

===2.2 归并排序===

归并,将两个有序数组归并成一个更大的有序数组。

归并排序,能够保证任意长度为N的数组排序所需时间和NlogN成正比,缺点是所需的额外空间与N成正比。

实现归并直接的方法,将两个不同的有序数组归并到第三个数组中,创建一个适当大小的数组,然后将两个输入数组的元素一个个从小到大放入这个数组中。

归并一个大数组时,需进行多次归并,每次归并都要创建一个新数组存储排序结果。通过原地归并,先将前半部分排序,再将后半部分排序,在数组中移动元素而不需要使用额外的空间。

merge(a, lo, mid, hi)它会将子数组a[lo...mid]和a[mid+1..hi]归并成一个有序数组,并将结果存放在a[lo..hi],

原地归并的抽象方法:

public static void merge(Comparable[] a, int lo, int mid, int hi){

int i = lo, j+mid+1;

for(int k =lo; k <= hi; k++) aux[k] = a[k];

for(int k =lo; k<=hi; k++)

if(i>mid) a[k]=aux[j++];

else if(j >hi) a[k]=aux[i++];

else if(less(aux[j], aux[i])) a[k] =aux[j++];

else a[k]=aux[i++];

}

该方法先将所有元素复制到aux[], 然后再归并回a[],方法再归并时进行了4个条件判断,左半边用尽(取右半边元素),右半边用尽(取左半边元素),右半边的当前元素小于左半边的当前元素(取右半边的元素),以及右半边的当前元素大于等于左半边的当前元素(取左半边的元素)

自顶向下的归并排序

public class Merge{

private static Comparable[] aux;

public static void sor(Comparable[] a){

aux = new Comparable[a.length];

sort(a, a.length-1);

}

private static void sort(Comparable[] a, int lo, int hi){

if(hi<=lo) return;

int mid = lo+(hi-lo)/2;

sort(a, lo, mid);

sort(a, mid+1, hi);

merge(a, lo, mid, hi);

}

}

自底向上的归并排序

public class MergeBu{

private static Comparable[] aux;

public static void sort(Comparable[] a){

int N =a.length;

aux = new Comparable[N];

for(int sz =1; sz<N; sz =sz+sz){

for(int lo=0; lo<N-sz; lo+=sz+sz)

merge(a, lo, lo+sz-1, Math.min(lo+sz+sz-1, N-1));

}

}

}

===2.3快速排序===

快速排序时间复杂度(O(nlogn));

特点:原地排序(只需要一个很小的辅助栈),且将长度为N的数组排序所需的时间和NlgN成正比。

快速排序:分治的排序算法,将一个数组分成两个子数组,独立的排序。与归并排序互补,

并排序将数组分成两个子数组分别排序,并将有序的子数组归并以将整个数组排序;而快速排序将数组排序的方式则是当两个子数组都有序时整个数组也就自然有序了。在第一种情况中,递归调用发生在处理整个数组之前;在第二种情况中,递归调用发生在处理整个数组之后。

在归并排序中,一个数组被等分为两半;在快速排序中,切分(partition)的位置取决于数组的内容

public class Quick

{ public static void sort(Comparable[] a) {

StdRandom.shuffle(a)

; // 消除对输入的依赖 sort(a, 0, a.length -1); }

private static void sort(Comparable[] a, int lo, int hi)

{ if (hi <= lo) return; int j = partition(a, lo, hi); // 切分(请见“快速排序的切分”)

sort(a, lo, j-1); // 将左半部分a[lo .. j-1]排序

sort(a, j+1, hi); // 将右半部分a[j+1 .. hi]排序

} }

该方法的关键在于切分,这个过程使得数组满足下面三个条件:❏对于某个j, a[j]已经排定;❏ a[lo]到a[j-1]中的所有元素都不大于a[j];❏ a[j+1]到a[hi]中的所有元素都不小于a[j]。通过递归地调用切分来排序的。

因为切分过程总是能排定一个元素,用归纳法不难证明递归能够正确地将数组排序:如果左子数组和右子数组都是有序的,那么由左子数组(有序且没有任何元素大于切分元素)、切分元素和右子数组(有序且没有任何元素小于切分元素)组成的结果数组也一定是有序的。

需要实现切分方法。一般策略是先随意地取a[lo]作为切分元素,即那个将会被排定的元素,然后我们从数组的左端开始向右扫描直到找到一个大于等于它的元素,再从数组的右端开始向左扫描直到找到一个小于等于它的元素。这两个元素显然是没有排定的,因此我们交换它们的位置。

当两个指针相遇时,我们只需要将切分元素a[lo]和左子数组最右侧的元素(a[j])交换然后返回j即可

快速排序的最好情况是每次都正好能将数组对半分。在这种情况下快速排序所用的比较次数正好满足分治递归的CN=2CN/2+N公式。2CN/2表示将两个子数组排序的成本,N表示用切分元素和所有数组元素进行比较的成本

几个细节问题:原地切分、别越界、保持随机性、终止循环、处理切分元素值有重复的情况、终止递归

算法改进:

切换到插入排序(对于小数组,快速排序比插入排序慢;因为递归,快速排序的sort()方法在小数组中也会调用自己);

改进快速排序性能的第二个办法是使用子数组的一小部分元素的中位数来切分数组。这样做得到的切分更好,但代价是需要计算中位数。人们发现将取样大小设为3并用大小居中的元素切分的效果最好;

熵最优的排序,含有大量重复元素的数组,一个元素全部重复的子数组就不需要继续排序了,但我们的算法还会继续将它切分为更小的数组,一个简单的想法是将数组切分为三部分,分别对应小于、等于和大于切分元素的数组元素。这种切分实现起来比我们目前使用的二分法更复杂

==2.4优先队列===

许多应用程序都需要处理有序的元素,但不一定要求它们全部有序,或是不一定要一次就将它们排序。很多情况下我们会收集一些元素,处理当前键值最大的元素,然后再收集更多的元素,再处理当前键值最大的元素

例如,你可能有一台能够同时运行多个应用程序的电脑(或者手机)。这是通过为每个应用程序的事件分配一个优先级,并总是处理下一个优先级最高的事件来实现的。例如,绝大多数手机分配给来电的优先级都会比游戏程序的高。

在这种情况下,一个合适的数据结构应该支持两种操作:删除最大元素和插入元素。这种数据类型叫做优先队列。优先队列的使用和队列(删除最老的元素)以及栈(删除最新的元素)类似。

基于二叉堆数据结构的一种优先队列的经典实现方法,用数组保存元素并按照一定条件排序,以实现高效地(对数级别的)删除最大元素和插入元素操作。

优先队列的一些重要的应用场景包括模拟系统,其中事件的键即为发生的时间,而系统需要按照时间顺序处理所有事件;任务调度,其中键值对应的优先级决定了应该首先执行哪些任务;

泛型优先队列的API,优先队列是一种抽象数据类型(请见1.2节),它表示了一组值和对这些值的操作。优先队列最重要的操作就是删除最大元素和插入元素,删除最大元素的方法名为delMax(),插入元素的方法名为insert()

优先队列的调用示例:考虑以下问题:输入N个字符串,每个字符串都对应着一个整数,你的任务就是从中找出最大的(或是最小的)M个整数(及其关联的字符串)。要我们能够高效地实现insert()和delMin(),下面的优先队列用例中调用了MinPQ的TopM就能使用优先队列解决这个问题。

初级实现:数组实现(无序)、数组实现(有序)、链表表示法

堆的定义:数据结构二叉堆能够很好地实现优先队列的基本操作。在二叉堆的数组中,每个元素都要保证大于等于另两个特定位置的元素。相应地,这些位置的元素又至少要大于等于数组中的另两个元素。

当一棵二叉树的每个结点都大于等于它的两个子结点时,它被称为堆有序。

如果我们用指针来表示堆有序的二叉树,那么每个元素都需要三个指针来找到它的上下结点(父结点和两个子结点各需要一个)。

由下至上的堆有序化(上浮)、由上至下的堆有序化(下沉)、多叉堆、调整数组大小、元素的不可变性、索引优先队列、索引优先队列用例。

用基于堆的优先队列这样做等同于哪种排序?一种全新的排序方法!下面我们就用堆来实现一种经典而优雅的排序算法——堆排序。

堆排序可以分为两个阶段。在堆的构造阶段中,我们将原始数组重新组织安排进一个堆中;然后在下沉排序阶段,我们从堆中按递减顺序取出所有元素并得到排序结果。

从右至左用sink()函数构造子堆。数组的每个位置都已经是一个子堆的根结点了,sink()对于这些子堆也适用。如果一个结点的两个子结点都已经是堆了,那么在该结点上调用sink()可以将它们变成一个堆。这个过程会递归地建立起堆的秩序。开始时我们只需要扫描数组中的一半元素,因为我们可以跳过大小为1的子堆。最后我们在位置1上调用sink()方法,扫描结束。

堆的构造、下沉排序、先下沉后上浮。

堆排序在排序复杂性的研究中有着重要的地位,因为它是我们所知的唯一能够同时最优地利用空间和时间的方法——在最坏的情况下它也能保证使用~2NlgN次比较和恒定的额外空间。

2.5 应用

排序如此有用的一个主要原因是,在一个有序的数组中查找一个元素要比在一个无序的数组中查找简单得多。只要队列是有序的,很多其他任务也更容易完成。

排序算法的一种典型应用就是商业数据处理。

我们在类的定义中实现一个恰当的compareTo()方法就可以做到这一点。这样我们在处理Transaction类型的数组a[]时就可以先将其排序,比如这样Quick.sort(a)。我们的排序算法对Transaction类型一无所知,但Java的Comparable接口使我们可以为该类型定义大小关系,这样我们的任意排序算法都能够用于Transaction对象了。

Java系统库的排序算法,这里我们考虑Java系统库中的主要排序方法java.util. Arrays.sort()。

ava的系统程序员选择对原始数据类型使用(三向切分的)快速排序,对引用类型使用归并排序。这些选择实际上也暗示着用速度和空间(对于原始数据类型)来换取稳定性(对于引用类型)。

排序应用一览:商业计算、信息搜索(有序的信息确保我们可以用经典的二分查找法(见第1章)来进行高效的搜索)、运筹学、事件驱动模拟、数值计算、组合搜索

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

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

相关文章

2024年R1快开门式压力容器操作证模拟考试题库及R1快开门式压力容器操作理论考试试题

题库来源&#xff1a;安全生产模拟考试一点通公众号小程序 2024年R1快开门式压力容器操作证模拟考试题库及R1快开门式压力容器操作理论考试试题是由安全生产模拟考试一点通提供&#xff0c;R1快开门式压力容器操作证模拟考试题库是根据R1快开门式压力容器操作最新版教材&#…

【IPC通信--消息队列】

消息队列&#xff08;也叫做报文队列&#xff09;是一个消息的链表。可以把消息看作一个记录&#xff0c;具有特定的格式以及特定的优先级。对消息队列有写权限的进程可以向消息队列中按照一定的规则添加新消息&#xff1b;对消息队列有读权限的进程则可以从消息队列中读走消息…

优秀案例 | 嘉吉动物营养虚拟人IP“小嘉”, 虚拟动力提供常态化高效率短视频制作工具

在流量见顶的时代 品牌宣传逐渐精细化 塑造一个具备亲和力及创新感的虚拟IP 可以持续扩大品牌影响力 与挖掘品牌更多可能性 「嘉吉动物营养」紧随营销趋势&#xff0c;通过广州虚拟动力「虚拟人运营套装」&#xff0c;将虚拟人IP运营与品牌宣传相结合&#xff0c;带动品牌形…

从音乐“卷”到直播,涨价也救不了腾讯音乐

继6月大规模涨价之后&#xff0c;腾讯音乐娱乐集团&#xff08;下称“腾讯音乐”&#xff0c;01698.HK&#xff09;旗下QQ音乐会员再次涨价。 「不二研究」据腾讯音乐三季报发现&#xff1a;在会员数这一关键指标上&#xff0c;腾讯音乐在三季度的月活跃用户从去年同期的6.20亿…

STM32深入系列02——BootLoader分析与实现

文章目录 1. STM32程序升级方法1.1 ST-Link / J-link下载1.2 ISP&#xff08;In System Programing&#xff09;1.3 IAP&#xff08;In Applicating Programing&#xff09;1.3.1 正常程序运行流程1.3.2 有IAP时程序运行流程 2. STM32 Bootloader实现2.1 方式一&#xff1a;Boo…

开启Android学习之旅-2-架构组件实现数据列表及添加(kotlin)

Android Jetpack 体验-官方codelab 1. 实现功能 使用 Jetpack 架构组件 Room、ViewModel 和 LiveData 设计应用&#xff1b;从sqlite获取、保存、删除数据&#xff1b;sqlite数据预填充功能&#xff1b;使用 RecyclerView 展示数据列表&#xff1b; 2. 使用架构组件 架构组…

HarmonyOS4.0系统性深入开发16进程模型概述

进程模型概述 HarmonyOS的进程模型&#xff1a; 应用中&#xff08;同一包名&#xff09;的所有UIAbility运行在同一个独立进程中。WebView拥有独立的渲染进程。 基于HarmonyOS的进程模型&#xff0c;系统提供了公共事件机制用于一对多的通信场景&#xff0c;公共事件发布者…

「网络安全术语解读」SARIF详解

引言&#xff1a;什么是SARIF&#xff1f;它的产生背景是什么&#xff1f;SARIF主要包含哪些内容&#xff1f;使用SARIF有哪些好处&#xff1f; 1. SARIF简介 SARIF&#xff08;Static Analysis Results Interchange Format &#xff0c;静态分析结果交换格式&#xff09;是一…

PTA——猴子吃桃问题

一只猴子第一天摘下若干个桃子&#xff0c;当即吃了一半&#xff0c;还不过瘾&#xff0c;又多吃了一个&#xff1b;第二天早上又将剩下的桃子吃掉一半&#xff0c;又多吃了一个。以后每天早上都吃了前一天剩下的一半加一个。到第N天早上想再吃时&#xff0c;见只剩下一个桃子了…

BOM介绍

文章目录 1、简介主要作用 2、BOM的组成2.1 窗口对象window2.1.1 window对象特点2.1.2 window作用域2.1.3 window对象常见方法**第一类&#xff1a;系统对话框**第二类&#xff1a;控制浏览器窗口方法第三类&#xff1a;与定时器有关的方法 1、简介 BOM&#xff08;Browser Ob…

docker安裝gocd-server,并配置gitlab授权登录

gocd的地址&#xff1a;Installing GoCD server on Windows | GoCD User Documentation gocd文档&#xff1a;GitHub - gocd/docker-gocd-server: Docker server image for GoCD 一、docker拉取gocd镜像 #拉取server镜像 docker pull gocd/gocd-server:v21.1.0docker pull g…

3的幂00

题目链接 3的幂 题目描述 注意点 无 解答思路 不断除以3直到除数或余数为0为止&#xff0c;判断除完后的数字是否为1 代码 class Solution {public boolean isPowerOfThree(int n) {while (n / 3 ! 0) {if (n % 3 ! 0) {return false;}n n / 3;}return n 1;} }关键点 …

springmvc内嵌tomcat、tomcat整合springmvc、自研国产web中间件

springmvc内嵌tomcat、tomcat整合springmvc、自研国产web中间件 这是由于公司老项目转化springboot存在太多坑&#xff0c;特别是hibernate事务一条就坑到跑路&#xff0c;你又不想搞没听说过的国产中间件兼容&#xff0c;又不想搞weblogic、WebSphere等中间件的适配&#xff…

【蓝桥杯软件赛 零基础备赛20周】第7周——二叉树

文章目录 1 二叉树概念2 二叉树的存储和编码2.1 二叉树的存储方法2.2 二叉树存储的编码实现2.3 二叉树的极简存储方法 3 例题4 习题 前面介绍的数据结构数组、队列、栈&#xff0c;都是线性的&#xff0c;它们存储数据的方式是把相同类型的数据按顺序一个接一个串在一起。简单的…

MPL3115A2大气压温度采集芯片的工作原理与特点详解

目录 一、引言 二、MPL3115A2主要特点和功能 三、主要优势 3.1 内部自动补偿 3.2 FIFO 四、硬件原理图 4.1 硬件连接 五、软件配置 六、资料获取 一、引言 MPL3115A2是一款高精度的大气压力传感器&#xff0c;能够测量大气压力、海拔高度和温度。它采用了MEMS&#xf…

Redis内存策略:「过期Key删除策略」+ 「内存淘汰策略」

Redis之所以性能强&#xff0c;最主要的原因就是基于内存存储&#xff0c;然而单节点的Redis其内存大小不宜过大&#xff0c;否则会影响持久化或主从同步的性能。 Redis内存满了&#xff0c;会发生什么&#xff1f; 在Redis的运行内存达到了某个阈值&#xff0c;就会触发内存…

Linux - No space left on device

问题描述 No space left on device 原因分析 说明在服务器设备上的存储空间已经满了&#xff0c;不能再上传或者新建文件夹或者文件等。 解决方案 确认查看服务器系统的磁盘使用情况是否是真的已经没有剩余空间&#xff0c;复制下面命令在服务器上运行&#xff0c;然后发现如果…

CSS 彩虹按钮效果

<template><view class"content"><button class"btn">彩虹按钮</button></view> </template><script></script><style>body{background-color: #000;}.content {margin-top: 300px;}.btn {width: 1…

jenkins忘记密码后的操作

1、先停止 jenkins 服务 systemctl stop jenkins 关闭Jenkins服务 或者杀掉进程 ps -ef | grep jenkins &#xff5c;awk {print $2} | grep -v "grep" | xargs kill -9 2、找到 config.xml 文件 find /root -name config.xml3、备份config.xml文件 cp /root/.jen…

添加一个编辑的小功能(PHP的Laravel)

一个编辑的按钮可以弹出会话框修改断更天数 前台 加一个编辑按钮的样式&#xff0c;他的名字是固定好的 之前有人封装过直接用就好&#xff0c;但是一定放在class里面&#xff0c;不要放在id里面 看见不认识的方法一定要去看里面封装的是什么 之前就是没有看&#xff0c;所以…