v8--sort方法源码中对于长度较短的数组使用的是插入排序法。

部分源码:

function InsertionSort(a, from, to) {
for (var i = from + 1; i < to; i++) {
var element = a[i];
// Pre-convert the element to a string for comparison if we know
// it will happen on each compare anyway.
var key =
(custom_compare || %_IsSmi(element)) ? element : ToString(element);
// place element in a[from..i[
// binary search
var min = from;
var max = i;
// The search interval is a[min..max[
while (min < max) {
var mid = min + ((max - min) >> 1);
var order = Compare(a[mid], key);
if (order == 0) {
min = max = mid;
break;
}
if (order < 0) {
min = mid + 1;
} else {
max = mid;
}
}
// place element at position min==max.
for (var j = i; j > min; j--) {
a[j] = a[j - 1];
}
a[min] = element;
}
}

源码的理解:

参数:a--数组。form起点索引即0。to终点索引。

源码对数组长度进行了计算,如值为undefined会被移至数组末尾并且长度会减去。

通常如果是正经的数组to就是a.length。

需要注意的是,以上都仅仅是在初始数组长度小于22时的情况。

此插入排序法是将待比较的数值与外层循环当前值对比,外层循环是从第二个数值开始的。

开始循环数组,索引初值为1。创建一个哨兵,保存当前循环索引对应的值 。

即key = a[i],该值需要插入到已循环数组的某个位置。

创建max,记录已循环次数。即 max = i

创建min,记录插入位置。初值是0 即 min = 0

内层循环1,目的找出插入的位置。循环条件 min < max

该循环内,创建mid 保存min和max的中值。然后计算该中值对应的数组值与哨兵的大小。即compare(arr[mid], key)

需要注意的是,mid min max 都是数组的索引。

如果计算结果为0,即说明索引中值对应值与哨兵值相等。

那么插入位置就是这个点。把mid赋值给min,取得插入点。退出该循环

如果计算值小于0。比如已经排序好的有序数组个数是15。索引中值是7。a[7]与key(即a[16])比较小于0。即key大于a[7]

说明key应该要插入到a[7]后面。则插入点最起码应该在索引7后面。故令min = 7 + 1;然后继续循环找出最合适的位置。

如果计算值大于0,还是上面那个例子。a[7]与key(即a[16])比较大于0。即key小于a[7]

说明key应该要插入到a[7]前面。则插入点最起码应该在索引7前面。故令max = mid;然后继续循环找出最合适的位置。

该循环一直到min = max后结束。取得min值。

接着以当前已循环次数i作为初始值,另存副本j = i,以j > min作为判断条件。

依次把前一索引值赋值给当前索引值,如 a[16] = a[15],a[15] = a[14] 等等等

最后把哨兵赋值给插入点。即a[min] = key

至于排序是为升降序。则由比较函数的判断

如最简单的比较函数:(x, y) => return x - y

返回参数1 减去 参数2 的值。即索引中值对应值减去哨兵

索引中值对应值小于哨兵,则计算结果小于0。按照上述逻辑,把mid+1赋值给min。则插入点在该索引后面,故为升序

索引中值对应值大于哨兵,则计算结果大于0。按照上述逻辑,把mid赋值给max。则插入点在该索引前面,还是升序

图解

代码:

function insertSort(arr) {

  for(let i= 1; i < arr.length; i++) {
let key = arr[i] // 哨兵,保存当前循环索引对应的值,需要插入到已循环数组中 let min = 0 // 插入位置
let max = i // 已循环次数 // 遍历已循环的数组,找寻插入位置
while(min < max) {
// 中值 >>有符号右移 类似于 Math.floor(int/2) int是正整数
let mid = min + ((max - min) >>1)
// 比较函数,传入索引中值对应的值和哨兵
const order = compare(a[mid], key) // 如果等于0
if (order === 0) {
// 把mid赋值给min。作为插入位置,退出循环
min = max = mid
break;
}
    // 小于0。插入位置在该索引中值后面
if (order < 0) {
min = mid + 1
} else { // 大于0。插入位置在该索引中值前面
max = mid
}
} for (let j = i; j > min; j--) {
arr[j] = arr[j - 1]
}
  // 插入
arr[min] = key
}
}

调试:

v8--sort 方法 源码 (1) 插入排序法的更多相关文章

  1. v8--sort 方法 源码 (2) 快速排序法

    v8 sort方法部分关于快速排序法的源码: function QuickSort(a, from, to) { // Insertion sort is faster for short array ...

  2. erlang下lists模块sort(排序)方法源码解析(二)

    上接erlang下lists模块sort(排序)方法源码解析(一),到目前为止,list列表已经被分割成N个列表,而且每个列表的元素是有序的(从大到小) 下面我们重点来看看mergel和rmergel ...

  3. erlang下lists模块sort(排序)方法源码解析(一)

    排序算法一直是各种语言最简单也是最复杂的算法,例如十大经典排序算法(动图演示)里面讲的那样 第一次看lists的sort方法的时候,蒙了,几百行的代码,我心想要这么复杂么(因为C语言的冒泡排序我记得不 ...

  4. Java split方法源码分析

    Java split方法源码分析 public String[] split(CharSequence input [, int limit]) { int index = 0; // 指针 bool ...

  5. getOrCreateEnvironment()方法源码探究

    该方法目的是创建一个环境对象,并且根据环境类型,自动判断是创建web环境对象,还是标准非web环境对象. 首先该方法源于prepareEnvironment准备环境: 然后进入该方法源码: 可以发现: ...

  6. TreeSet集合的add()方法源码解析(01.Integer自然排序)

    >TreeSet集合使用实例 >TreeSet集合的红黑树 存储与取出(图) >TreeSet的add()方法源码     TreeSet集合使用实例 package cn.itca ...

  7. invalidate和requestLayout方法源码分析

    invalidate方法源码分析 在之前分析View的绘制流程中,最后都有调用一个叫invalidate的方法,这个方法是啥玩意?我们来看一下View类中invalidate系列方法的源码(ViewG ...

  8. Linq分组操作之GroupBy,GroupJoin扩展方法源码分析

    Linq分组操作之GroupBy,GroupJoin扩展方法源码分析 一. GroupBy 解释: 根据指定的键选择器函数对序列中的元素进行分组,并且从每个组及其键中创建结果值. 查询表达式: var ...

  9. Java线程状态、线程start方法源码、多线程、Java线程池、如何停止一个线程

    下面将依次介绍: 1. 线程状态.Java线程状态和线程池状态 2. start方法源码 3. 什么是线程池? 4. 线程池的工作原理和使用线程池的好处 5. ThreadPoolExecutor中的 ...

随机推荐

  1. NIO通信中connect()方法和finishConnect()方法的区别

    1.对于阻塞模式下,调用connect()进行连接操作时,会一直阻塞到连接建立完成(无连接异常的情况下).所以可以不用finishConnect来确认. 2.但在非阻塞模式下,connect()操作是 ...

  2. odoo开发笔记 -- 触发机制/埋点设置

    场景描述: 项目需求中,经常会需要,当某个字段处某个特定状态时候,触发执行特定的方法:或者创建某条记录的时候,同时做另一个操作:如何实现类似的需求? 实现方式: odoo中提供了几种触发方式: 1. ...

  3. 解决 No IDEA annotations attached to the JDK 1.8和xml文件没有代码提示

    Android studio3.3 用着用着突然xml里没有代码联想了,忙着做其他的就没管,写xml的时候就硬写... 然后今天用着突然在class文件上方提示No IDEA annotations ...

  4. MongoDB开发深入之三:复制

    复制是基于操作日志oplog,相当于MySQL中的二进制日志,只记录发生改变的记录,复制是将主节点的oplog日志同步并应用到其他从节点的过程. 首先要理解两个概念:1.复制:提供冗余和高可用性:2. ...

  5. sqlite3 数据库表查看步骤

    sqlite3 数据库表查看步骤 1 sqlite3 local.db2 .mode column3 .headers on4 .tables5 select * from tablename6 ;( ...

  6. [Sw] Swoole 生态迷局,基于 Swoole 的第 109 框架

    这两天,又一全栈式 Swoole 协程框架面世了 - hyperf,实现思路是我内心点了赞同的,就集成现有 PHP 生态优质组件到 Swoole 的协程中来. 有人想到,为什么不是 Swoole 集成 ...

  7. [UE4] TSharedPtr, TWeakObjectPtr and TUniquePtr

    转自:https://dawnarc.com/2018/07/ue4-tsharedptr-tweakobjectptr-and-tuniqueptr/ UE4 的 TSharedPtr.TWeakO ...

  8. Hadoop window win10 基础环境搭建(2.8.1)(转)

    下面运行步骤除了配置文件有部分改动,其他都是参照hadoop下载解压的share/doc/index.html. hadoop下载:http://apache.opencas.org/hadoop/c ...

  9. Git 克隆

    工作中大家分工协作,共同开发维护一个项目,git仓库放在公共的服务器上,如github.Gitee等. 在这种情况使用git,需要先克隆仓库到本地. 克隆非常简单,使用命令git clone 将远程仓 ...

  10. JS系列:数据类型详细讲解

    ctrl+B加粗 ### 数据类型: - 1.number数字类型 NaN:not a number 不是一个数 ,但他是数字类型 isNaN:检测当前值是否不是有效数字,返回true代表不是有效数字 ...