[读书笔记]算法(Sedgewick著)·第二章.初级排序算法
本章开始学习排序算法

1.初级排序算法
先从选择排序和插入排序这两个简单的算法开始学习排序算法。选择排序就是依次找到当前数组中最小的元素,将其和第一个元素交换位置,直到整个数组有序。
public static void sort(Comparable a[]){
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);
}
}
而插入排序就像打poker一样,找到每个元素在小数组中的位置。
public static void sort(Comparable a[]){
int N = a.length;
for(int i = 1; i < N; i ++){
//将a[i]插入a[i-1]、a[i-2]、a[i-3]...之中
for(int j = i; j > 0 && less(a[j], a[j-1]); j--){
exch(a, j, j-1);
}
}
}
算法思路相当简单,而本节比较有趣的地方是可视化排序算法。我们就实现这个可视化。
我们分为三个步骤。第一步,将一个数组的元素(这里我们选用double数据类型)用长条表示,并依次罗列在屏幕上。第二步,按照各个算法的思路将有序数组排序,画出一张可视轨迹图。第三步,将轨迹图动画化。
接下来的重头戏是比较选择排序和插入排序。通过下图的例子,我们可以看出第i个元素都进行一次交换和N-1-i次比较。对于长度为N的数组,选择排序要进行N^2/2次比较和N次交换。

而对于插入排序呢?情况比选择排序复杂。最坏情况(完全逆序)下,需要~N^2/2次比较和~N^2/2次交换;而最好情况(已经顺序)下,需要N-1次比较和0次交换。就如同例子中第3次插入时,S、T已经是有序的了,所以不存在交换的需要。对于随机排序的数组,在平均情况下每个元素都可能向后移动半个数组的长度,因此交换总数是~N^2/4,而比较次数是交换次数加上一个额外的项(相比N^2,可以忽略)。

而排序前数组的倒置(比如E-A)个数越少,插入排序就越快。接下来我们用随机数组来比较两种算法。
重复一百次的1000个随机数排序比较结果:

重复一百次的10000个随机数排序比较结果:

最后,我们可以得出:对于随机数组,插入排序和选择排序的运行时间是平方级别的,两者之比是一个较小的常数(本机测试为1.3~1.4)。
希尔排序的思想是使数组中任意减个为h的元素有序,也就是将插入排序的元素移动距离由1改变为逐渐增大的h。
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;
}
}

这么做的原因是因为部分有序数组更适合插入排序。那么,我们就比较一下这两种算法的执行速度。

[读书笔记]算法(Sedgewick著)·第二章.初级排序算法的更多相关文章
- 第二章:排序算法 及其他 Java代码实现
目录 第二章:排序算法 及其他 Java代码实现 插入排序 归并排序 选择排序算法 冒泡排序 查找算法 习题 2.3.7 第二章:排序算法 及其他 Java代码实现 --算法导论(Introducti ...
- Python基础教程 读书笔记(2)第二章 列表和元组
2.1序列概览 列表和元组的主要区别在于,列表可以修改,元组则不能.也就是说如果要根据要求来添加元素,那么列表可能会更好用;而出于某些原因,序列不能修改的时候,使用元组则更为合适.使用后者的理由通常是 ...
- 读书笔记 - javascript 高级程序设计 - 第二章 在Html中使用JavaScript
1 <script>的6个属性 async 立即下载当前script标签的外部脚本 但不能影响别的 charset 没用了 defer 文档显示之后再执行脚本,只对外部脚本有效 lan ...
- 《算法导论》第二章demo代码实现(Java版)
<算法导论>第二章demo代码实现(Java版) 前言 表示晚上心里有些不宁静,所以就写一篇博客,来缓缓.囧 拜读<算法导论>这样的神作,当然要做一些练习啦.除了练习题与思考题 ...
- 《算法4》2.1 - 选择排序算法(Selection Sort), Python实现
选择排序算法(Selection Sort)是排序算法的一种初级算法.虽然比较简单,但是基础,理解了有助于后面学习更高深算法,勿以勿小而不为. 排序算法的语言描述: 给定一组物体,根据他们的某种可量化 ...
- JavaScript 数据结构与算法之美 - 十大经典排序算法汇总(图文并茂)
1. 前言 算法为王. 想学好前端,先练好内功,内功不行,就算招式练的再花哨,终究成不了高手:只有内功深厚者,前端之路才会走得更远. 笔者写的 JavaScript 数据结构与算法之美 系列用的语言是 ...
- [读书笔记]算法(Sedgewick著)·第一章(1)
到家放松放松之后就开始学习算法了,手里拿的是拿的是一本Robert Sedgewick的橙皮书<算法(第四版)>的.这本书与导论那本书的不同之处在于轻数学思想.重实现,也就是说这是一本很不 ...
- [读书笔记]算法(Sedgewick著)·第一章(2)
接着上一篇,mindmap更新如下内容. 3.背包.队列和栈 这节主要讲述了这三种数据结构(Bag.Queue.Stack)的API.实现以及链表.Queue和Stack还含有删除元素的方法.并引出了 ...
- Unity Shader入门精要读书笔记(一)序章
本系列的博文是笔者读<Unity Shader入门精要>的读书笔记,这本书的章节框架是: 第一章:着手准备. 第二章:GPU流水线. 第三章:Shader基本语法. 第四章:Shader数 ...
随机推荐
- vim emmet配置
http://nerd-is.in/2013-12/learning-vim-again-1-install-vundle/ http://nerd-is.in/2013-12/learn-vim-a ...
- BZOJ 1227 虔诚的墓主人
Description 小W 是一片新造公墓的管理人.公墓可以看成一块N×M 的矩形,矩形的每个格点,要么种着一棵常青树,要么是一块还没有归属的墓地.当地的居民都是非常虔诚的基督徒,他们愿意提前为自己 ...
- C#<热血传奇>服务端源代码再次给力更新
前段时间一直在忙公司项目,最近抽点空稍微把部分代码重新整理一下(代码太久没碰很多地方都忘记了 囧~~~) 此次还是更新服务端,并修正上一版里面存在的很多指针 异常BUG... (代码比较烂,还请各位 ...
- 使用js为html元素动态添加class
<ul id="root"> <li>1</li> <li>2</li> <li>3</li> ...
- 实现SELECT的全选,反选,AB选的JAVASCRIPT代码
参考网上,用原生JS粗糙实现. 我发现用UIKIT的BUTTON会自动刷新我那核心的模态窗口,只好用另外的LABEL或CODE标签了. $(".btn-select-all").c ...
- android gridview按钮边框和定制点击颜色
<?xml version="1.0" encoding="utf-8"?> <GridView xmlns:android="ht ...
- Codeforces Round #238 (Div. 1)
感觉这场题目有种似曾相识感觉,C题还没看,日后补上.一定要坚持做下去. A Unusual Product 题意: 给定一个n*n的01矩阵,3种操作, 1 i 将第i行翻转 2 i 将第i列翻转 3 ...
- SQLite入门与分析(八)---存储模型(3)
写在前面:接上一节,本节主要讨论索引页面格式,以及索引与查询优化的关系. (1)索引页面格式sqlite> select * from sqlite_master;table|episodes| ...
- Android 关于HttpClient上传中文乱码的解决办法
使用过HttpClient的人都知道可以通过addTextBody方法来添加要上传的文本信息,但是,如果要上传中文的话,或还有中文名称的文件会出现乱码的问题,解决办法其实很简单: 第一步:设置Mult ...
- Android开发中一些被冷落但却很有用的类和方法
MediaMetadataRetriever 顾名思义,就是用来获取媒体文件一些相关信息的类.包括一首歌的标题,作者,专辑封面和名称,时长,比特率等等.如果是视频的话,可以获取视频的长宽,预览图. h ...