js内置的Array函数原型对象有个sort方法,这个方法能按照顺序排序数组。

例如:

 var arr1 = [6, 4, 2, 5, 2];
arr1.sort((x, y) => x - y);
console.log(arr1); // [2, 2, 4, 5, 6];

以升序为例,这个方法的实现原理的简单理解:

第一轮比较。

先拿第一个数和第二个数字比较,如果第一个比第二个大,则交换位置。

接着又拿第一个数和第三个数比较,如果第一个比第三个大,则交换位置。

。。。

最后拿第一个数与最后一个数比较,如果第一个数比最后一个数大,则交换位置。

以上是第一轮比较。经过第一轮的比较。第一个数,就是数组中值最小的数了。

用上述例子来描述第一轮比较的过程,如图:

接着开始第二轮比较

先拿第二个数与第三个数比较,如果第二个比第三个大,则交换位置。

然后又拿第二个数和第四个数比较,如果第二个比第四个大,则交换位置。

。。。

最后拿第二个数与最后一个数比较,如果第二个数比最后一个数大,则交换位置。

以上是第二轮比较。经过第二轮的比较。第二个数,就是数组中值的第二小的数了。

用上述例子来描述第二轮比较的过程,如图:

循环往复直至全部都比较完。

上述例子的比较过程的代码演示:

 function mySort(arr, cb) {
window.count = 0;// 统计循环次数
const res = cb && cb(1, 2);
let i, j, len = arr.length;
if (res < 0) {// 升序
for (i = 0; i < len; i++) {
for (j = i + 1; j < len; j++) {
count++;
if (arr[i] > arr[j]) {
[arr[i], arr[j]] = [arr[j], arr[i]];
}
}
}
} else {// 降序
for (i = 0; i < len; i++) {
for (j = i + 1; j < len; j++) {
count++;
if (arr[i] < arr[j]) {
[arr[i], arr[j]] = [arr[j], arr[i]];
}
}
}
}
}

这种排序方法循环次数为 (len - 1) * len / 2。

验证:

 const arr = [];
for (let i = 0; i < 10; i++) {
arr.push(Math.floor(Math.random()*100));
}
mySort(arr, (x, y) => x - y);

结果:

采用排序二叉树结构的算法。代码演示:

 function mySort(arr, cb) {
const res = cb && cb(1 ,2);
class Node {
constructor(key) {
this.key = key;
this.left = null;
this.right = null;
}
};
class BinaryTree {
constructor(key) {
this.root = null;
this.init(key);
}
init(key) {
this.root = new Node(key);
}
insert(key) {
const newNode = new Node(key);
this._insertNode(this.root, newNode);
}
_insertNode(root, node) {
if (node.key < root.key) {
if (!root.left) {
root.left = node;
} else {
this._insertNode(root.left, node);
}
} else {
if (!root.right) {
root.right = node;
} else {
this._insertNode(root.right, node);
}
}
}
inorderTraversal(callback) {
if (res < 0) {
this._inorderTraversalNodeSmall(this.root, callback);
} else {
this._inorderTraversalNodeBig(this.root, callback);
}
}
_inorderTraversalNodeSmall(node, callback) {
if (!!node) {
this._inorderTraversalNodeSmall(node.left, callback);
callback(node);
this._inorderTraversalNodeSmall(node.right, callback);
}
}
_inorderTraversalNodeBig(node, callback) {
if (!!node) {
this._inorderTraversalNodeBig(node.right, callback);
callback(node);
this._inorderTraversalNodeBig(node.left, callback);
}
}
} const binaryTree = new BinaryTree(arr[0]);
let i, len;
for(i = 1, len = arr.length; i < len; i++) {
binaryTree.insert(arr[i]);
}
const _arr = [];
binaryTree.inorderTraversal(node => {
_arr.push(node.key);
});
return _arr;
}

测试:

 const arr = [];
for (let i = 0; i < 10; i++) {
arr.push(Math.floor(Math.random()*100));
}
const newArr1 = mySort(arr, (x, y) => y - x);
const newArr2 = mySort(arr, (x, y) => x - y);

结果:

当数组比较大的时候,后面这种算法的优势明显。

现以后面这种算法为例,测试代码:

 const arr = [];
for (let i = 0; i < 5000000; i++) {
arr.push(Math.floor(Math.random()*50000000));
}
const time = new Date().getMinutes() + ':' + new Date().getSeconds();
mySort(arr, (x, y) => x - y);
const nextTime = new Date().getMinutes() + ':' + new Date().getSeconds();

结果:

数组长度为5000000时,函数大概运行了6秒。

如果以第一种算法为例,5000000估计要好几分钟,这里缩减一下,设置成50000。测试代码:

 const arr = [];
for (let i = 0; i < 50000; i++) {
arr.push(Math.floor(Math.random()*50000000));
}
const time = new Date().getMinutes() + ':' + new Date().getSeconds();
mySort(arr, (x, y) => x - y);
const nextTime = new Date().getMinutes() + ':' + new Date().getSeconds();

结果:

数组长度为50000时,函数大概运行了10秒。

第一种算法的耗时貌似也不是平滑增加的。估计也是一条陡峭的曲线。数学知识都忘光了。以后搞明白了在来完善。

二叉树的实现原理的简单理解

javascript 数组排序原理的简单理解的更多相关文章

  1. mDNS 原理的简单理解

    转自:http://www.binkery.com/post/318.html mDNS 原理的简单理解 mDNS multicast DNS , 使用5353端口. 在局域网内,你要通过一台主机和其 ...

  2. mDNS原理的简单理解——每个进入局域网的主机,如果开启了mDNS服务的话,都会向局域网内的所有主机组播一个消息,我是谁,和我的IP地址是多少。然后其他也有该服务的主机就会响应,也会告诉你,它是谁,它的IP地址是多少

    MDNS协议介绍 mDNS multicast DNS , 使用5353端口,组播地址 224.0.0.251.在一个没有常规DNS服务器的小型网络内,可以使用mDNS来实现类似DNS的编程接口.包格 ...

  3. [转帖]mDNS原理的简单理解

    mDNS原理的简单理解 https://binkery.com/archives/318.html 发现还有avahi-daemon mdns 设置ip地址 等等事项 网络部分 自己学习的还是不够多 ...

  4. How Javascript works (Javascript工作原理) (五) 深入理解 WebSockets 和带有 SSE 机制的HTTP/2 以及正确的使用姿势

    个人总结: 1.长连接机制——分清Websocket,http2,SSE: 1)HTTP/2 引进了 Server Push 技术用来让服务器主动向客户端缓存发送数据.然而,它并不允许直接向客户端程序 ...

  5. HBase笔记:对HBase原理的简单理解

    早些时候学习hadoop的技术,我一直对里面两项技术倍感困惑,一个是zookeeper,一个就是Hbase了.现在有机会专职做大数据相关的项目,终于看到了HBase实战的项目,也因此有机会搞懂Hbas ...

  6. 一些JavaScript中原理的简单实现

    实现一个双向数据绑定 Object.defineProperty() 方法会直接在一个对象上定义一个新属性,或者修改一个对象的现有属性, 并返回这个对象.通过这个属性可以实现简单的双向数据绑定,当前版 ...

  7. 【原创】分布式之数据库和缓存双写一致性方案解析(三) 前端面试送命题(二)-callback,promise,generator,async-await JS的进阶技巧 前端面试送命题(一)-JS三座大山 Nodejs的运行原理-科普篇 优化设计提高sql类数据库的性能 简单理解token机制

    [原创]分布式之数据库和缓存双写一致性方案解析(三)   正文 博主本来觉得,<分布式之数据库和缓存双写一致性方案解析>,一文已经十分清晰.然而这一两天,有人在微信上私聊我,觉得应该要采用 ...

  8. 简单理解JavaScript闭包

    很多关于JS的书籍例如<JavaScript权威指南>或者<高程>都把闭包解释的晦涩难懂,萌新们是怎么也看不懂啊!不过别怕,今天我就用很简单的方式给大家讲解下到底什么是闭包.这 ...

  9. 深入理解JavaScript Hijacking原理

    最近在整理关于JavaScript代码安全方面的资料,在查关于JavaScript Hijacking的资料时,发现关于它的中文资料很少,故特意整理一下. 一.JavaScript Hijacking ...

随机推荐

  1. HDFS练习

    利用Shell命令与HDFS进行交互 以”./bin/dfs dfs”开头的Shell命令方式 1.目录操作 在HDFS中为hadoop用户创建一个用户目录(hadoop用户) 在用户目录下创建一个i ...

  2. 数据库sql优化总结之3--SQL优化总结

    SQL是每个Java程序员必回的一项技能,  对于项目中的各种复杂业务, 你是否能写出高效率, 简洁的SQL对于项目的运行效率和稳定性是有非常大的作用的. 通过个人的理解和网上的资料总结了一下常见的S ...

  3. Qt开发经验小技巧1-10

    当编译发现大量错误的时候,从第一个看起,一个一个的解决,不要急着去看下一个错误,往往后面的错误都是由于前面的错误引起的,第一个解决后很可能都解决了. 定时器是个好东西,学会好使用它,有时候用QTime ...

  4. (三)用CONCAT 函数 拼接字段

    一.将两个列拼接成一个列 数据源 select CONCAT(TRIM(username),'(',locaiton,')') from user2 解释: TRIM()函数用于去除字符串左右两边的空 ...

  5. JAVA Asponse.Word Office 操作神器,借助 word 模板生成 word 文档,并转化为 pdf,png 等多种格式的文件

    一,由于该 jar 包不是免费的, maven 仓库一般不会有,需要我们去官网下载并安装到本地 maven 仓库 1,用地址   https://www-evget-com/product/564  ...

  6. MySQL之备份

    MySQL备份和备份 备份/还原 冷备:需要停止当前正在运行mysqld,然后直接拷贝或打包数据文件. 半热备:mysqldump+binlog --适合数据量比较小的应用 在线热备:AB复制 --实 ...

  7. React+antd+less框架搭建步骤,看吧,整的明白儿的

    1.node版本 首先你要先看下你的node版本,如果小于10,建议升级到10及以上,因为低版本的 node 在自动创建 react框架时,有配置文件跟10及以上的有比较大的差异,而且需要增加.修改的 ...

  8. (转)面试前必知Redis面试题—缓存雪崩+穿透+缓存与数据库双写一致问题

    背景:redis问题在面试过程中经常被问到,对于常见问题一定不能放过. 面试前必知Redis面试题—缓存雪崩+穿透+缓存与数据库双写一致问题 一.缓存雪崩 1.1什么是缓存雪崩? 如果缓存数据设置的过 ...

  9. 08 Tomcat+Java Web项目的创建和War的生成

    1.web服务器软件:服务器:安装了服务器软件的计算机服务器软件:接收用户的请求,处理请求,做出响应 * web服务器软件:接收用户的请求,处理请求,做出响应. 在web服务器软件中,可以部署web项 ...

  10. 下载 m3u8 直播流的方法

    下载 FFmpeg http://ffmpeg.org/download.html 查找直播流地址 找到目标视频对应的 m3u8 播放列表. 执行脚本 ffmpeg -i https://nhkmov ...