受大学室友的鼓动,我也打算利用公众平台来记录自己的前端知识积累,同时呢,自己总结的东西,总归会有局限性,希望小伙伴能给我指点迷津。知识就是一张巨大的网,作为一名摸不清头绪的入学者,唯一能做的事情就是吐好每一根丝,丝拧成线,线再织成网。好啦,开机仪式over,废话不多说了啦...


关于Sort()这个函数,决定研究它是因为在看阮老师的箭头函数时,最后有一个小练习:
请使用箭头函数简化排序时传入的函数:

  1. var arr = [10, 20, 1, 2];
  2. arr.sort((x, y) => {
  3. ???
  4. });
  5. console.log(arr); // [1, 2, 10, 20]

因为之前js基础也不扎实,没有get到这个题的核心,想了想写了写,最后放弃了,看了评论里的答案。我的天,就一句——return x - y;,当时我就觉得这个函数太神奇了,这么简单的解决了数组排序。(上大学那会,懒惰致死的我所有排序算法原理都明明白白,但是从来没有写过,于是就...后悔莫及阿)

sort()函数定义

了解原理先从函数定义入手,于是乎...从W3C上搬了一段解释:

定义和用法: sort() 方法用于对数组的元素进行排序。

语法: arrayObject.sort(sortby)

返回值: 对数组的引用。请注意,数组在原数组上进行排序,不生成副本。

说明:

  1. 如果调用该方法时没有使用参数,将按字母顺序对数组中的元素进行排序,说得更精确点,是按照字符编码的顺序进行排序。要实现这一点,首先应把数组的元素都转换成字符串(如有必要),以便进行比较。
  2. 如果想按照其他标准进行排序,就需要提供比较函数,该函数要比较两个值,然后返回一个用于说明这两个值的相对顺序的数字。比较函数应该具有两个参数 a 和 b,其返回值如下:
    若 a 小于 b,在排序后的数组中 a 应该出现在 b 之前,则返回一个小于 0 的值。
    若 a 等于 b,则返回 0。
    若 a 大于 b,则返回一个大于 0 的值。

sort排序的过程中发生了什么呢

此处参考了前辈的文章链接描述,我就特别傻,断点打到sort()函数这一行,然后step-into执行,不断在console里打印arr。。。傻的一p
怎么查看sort()方法是如果实现排序的呢?我们可以在比较函数里把a,b及数组输出一下,看看是否能够看出使用的排序算法。

  1. var arr=[1, 8, 3, 5, -1];
  2. function compare(a,b){
  3. console.log(a,b,arr);
  4. return a-b;
  5. }
  6. arr.sort(compare);

控制台输出
1 8 [1, 8, 3, 5, -1]
8 3 [1, 8, 3, 5, -1]
1 3 [1, 8, 8, 5, -1]
8 5 [1, 3, 8, 5, -1]
3 5 [1, 3, 8, 8, -1]
8 -1 [1, 3, 5, 8, -1]
5 -1 [1, 3, 5, 8, 8]
3 -1 [1, 3, 5, 5, 8]
1 -1 [1, 3, 3, 5, 8]
[-1,1, 3, 5, 8]
*/
  第一次1和8比较,1<8,不需要调整位置。   

  第二次8和3比较,8>3,需要调整位置。但是这里没有交换位置,仅仅是8覆盖了3位置。这里就可以推断出不是单纯的使用了冒泡算法。
  第三是1和3比较,1<3,3替换了8的位置。什么鬼,几个意思???看到这里我也是表示不懂呀。那就继续往下看咯。   

  第四是8和5比较,8>5,又仅仅是覆盖,没有交换位置。还是不懂,继续往下!
  第五是3和5比较,3<5,5替换了8的位置,不懂,继续往下!   

  第六是8和-1比较,8>-1, 还仅仅是覆盖,继续往下!
  第七、八、九次,-1依次和5,3,1做了比较,并且5,3,1都移动了一次位置。

  我们得出了结论:sort()方法是使用的冒泡和插入两种方式结合进行排序的。

回顾冒泡和插入排序

这里我用自己的话总结一下:
冒泡排序:

  1. 第一轮:从第一个元素开始,相邻元素比较,前者比后者大,互换位置,下标加一;再继续相邻元素比较,大的元素移到后面,下标加一再比较...这样一轮比较下来,最后一个元素一定是数组里最大的元素
  2. 第二轮及之后:好啦,现在最后一个元素(即最大的元素)确定了,将其排除在外,我们再来从头对比除它之外的元素,将倒数第二大的元素移到倒数第二个位置。以此类推,每一轮确定一个元素的位置,就像小鱼吐泡泡一样,大的泡泡一点一点往上移
  1. function bubbel(arr) {
  2. var len=arr.length;
  3. for(var i=0; i&lt;len; i++) {
  4. for(var j=0; j&lt;len; j++) {
  5. if(arr[j] &gt; arr[j+1]) {
  6. var temp = arr[j];
  7. arr[j] = arr[j+1];
  8. arr[j+1] = temp;
  9. }
  10. }
  11. }
  12. return arr;
  13. }

插入排序:
1.第一轮:将第一个元素看成只有一个元素的有序数组,拿第二个元素和它比较,比它小就插到它前面,比它大就插到它后面。
2.第二轮:经过第一轮,前两个元素已为有序数组。再拿第三个元素和前两个元素比较,看插在哪合适。以此类推。一般会新建一个数组记录排序后的数组。

  1. // 插入排序 从下标1开始每增1项排序一次,越往后遍历次数越多
  2. function sort1(array) {
  3. var len = array.length,
  4. i, j, tmp, result;
  5. // 设置数组副本
  6. result = array.slice(0);
  7. for(i=1; i &lt; len; i++){
  8. tmp = result[i];
  9. j = i - 1;
  10. while(j&gt;=0 &amp;&amp; tmp &lt; result[j]){
  11. result[j+1] = result[j];
  12. j--;
  13. }
  14. result[j+1] = tmp;
  15. }
  16. return result;
  17. }

sort()的正面目来啦

前面铺垫了这么多,终于到了今天的重点——冒泡排序和插入排序是如何混合使用的?即sort()实现的原理。先上我的代码!! 时隔多年,我终于不再懒惰,勇敢写出我的代码。希望努力不要太晚~

  1. var arr = [1, 8, 3, 5, -1];
  2. var len = arr.length;
  3. function compareSet(temp, compare_i){
  4. for(var i = compare_i; i &gt; 0; i--){
  5. if(temp &gt; arr[i-1]){
  6. arr[i] = temp;
  7. break;
  8. }
  9. else{
  10. arr[i] = arr[i-1];
  11. arr[i-1] = temp;
  12. }
  13. }
  14. }
  15. for(var i = 0; i &lt; len; i++){
  16. if(arr[i] &gt; arr[i+1]){
  17. var temp = arr[i+1];
  18. arr[i+1] = arr[i];
  19. console.log(arr);
  20. compareSet(temp, i);
  21. }
  22. }
  23. console.log(arr);

根据之前分析sort()排序控制台输出,先是像冒泡排序那样相邻的元素比较。但是,一旦出现需要换位置的操作时,不再是像插入排序那样直接交换。而是先用变量temp暂存arr[i+1],再将较大的arr[i]移到[i+1]位置上,对暂存变量temp使用插入排序,将其插入前0 ~ [i-1]有序数组中。把这个temp安排好,再继续之前的冒泡排序。
**冒泡排序是元素对调后这一轮就不管事了,要重复i-1轮冒泡。
插入排序是不管现有元素的顺序是否正确,都给你在已有序数组从头比较到尾。**
所以,混合起来666。

这里还有一个前辈写的sort()实现,我对比一下,我的运行速度18ms,前辈的25ms。其实我感觉我写的没有前辈的简洁,但不知道为什么我的快一些。之后再仔细研究研究。

  1. [function findMinIndex(arr,start){
  2. var iMin=arr\[start\];
  3. var iMinIndex=start;
  4. for(var i=start;i&lt;arr.length;i++){
  5. if(iMin&gt;arr\[i\]){
  6. iMin=arr\[i\];
  7. iMinIndex=i;
  8. }
  9. }
  10. return iMinIndex;
  11. }
  12. for(var i=0;i&lt;arr.length;i++){
  13. var n=findMinIndex(arr,i);
  14. var tem;
  15. tem=arr\[n\];
  16. arr\[n\]=arr\[i\];
  17. arr\[i\]=tem;
  18. }][1]

其实,写到这里,应该结束了!但是我忽然想起来,大学《数据结构》课上,我最喜欢的魏莱老师好像给我们说过这个混合排序算法的。老师一步步引导我们思考的场景还历历在目,我甚至都可以回想起老师说话时的语气。可是,我却还的差不多了,实在是可恶!!!

原文地址:https://segmentfault.com/a/1190000017026510

小试牛刀之sort()排序的实现的更多相关文章

  1. 2.sort 排序命令讲解

    sort命令  sort:文本排序,仅仅是对显示文件的排序,而不影响源文件的顺序,是根据ASSII码     的字符升序来排列的.        -n:安装数值大小从小到大排列 ,默认是升序.     ...

  2. 反向输出及sort排序

    建立条件:#include "algorithm"引用这个头文件 1.reverse 的用法,反向排序,由自己输入5个数: 1 2 3 4 5 for (int i = 0; i ...

  3. JAVA Collections工具类sort()排序方法

    主要分析内容: 一.Collections工具类两种sort()方法 二.示例 一.Collections工具类两种sort()方法 格式一: public static <T extends ...

  4. javascript:算法之数组sort排序

    数组sort排序 sort比较次数,sort用法,sort常用 描述 方法sort()将在原数组上对数组元素进行排序,即排序时不创建新的数组副本.如果调用方法sort()时没有使用参数,将按字母顺序( ...

  5. sort排序

    /*问题 L: 使用sort排序题目描述标准库的sort函数给我们提供了一个很方便的排序的方法,光听别人说方便不顶事,得自己亲自实践一下才能体会到它的方便之处. 输入每组包含多组数据,每组数据第一行包 ...

  6. [转] C++的STL库,vector sort排序时间复杂度 及常见容器比较

    http://www.169it.com/article/3215620760.html http://www.cnblogs.com/sharpfeng/archive/2012/09/18/269 ...

  7. List<T>.Sort() 排序的用法

    List<T> 可以通过 .Sort()进行排序,但是当 T 对象为自定义类型时(比如自定义模型),就需要 IComparable接口重写其中的方法来实现,实现代码如下: class Pr ...

  8. sort排序中的坑

    问题的产生原因: 在一篇阿里面试题的跟帖中,很多人应用sort()方法对数组进行排序.看似合情合理的代码,运行结果却频频出错.为什么呢?因为很多人都忽略掉了一点,那就是sort()排序默认情况下是按A ...

  9. STL vector+sort排序和multiset/multimap排序比较

    由 www.169it.com 搜集整理 在C++的STL库中,要实现排序可以通过将所有元素保存到vector中,然后通过sort算法来排序,也可以通过multimap实现在插入元素的时候进行排序.在 ...

随机推荐

  1. zabbix监控websphere的几个监控项

    首先,我要吐槽一下这个AIX系统,这该死的天杀的玩个锤子象拔蚌的系统,没有自动补齐,删除文本字符也跟linux不一样,这让用惯的linux的我各种蓝瘦. 这个问题是在项目中遇到的,由于没有接触过AIX ...

  2. Problem 6

    Problem 6 # Problem_6.py """ The sum of the squares of the first ten natural numbers ...

  3. 使用maven创建springMVC时返回页面报错

    这是由于你的 Maven 编译级别是 jdk1.5 或以下,而你导入了 jdk1.6 以上的依赖包 解决办法: <build> <finalName></finalNam ...

  4. Java 内部类机制

    内部类(inner class)是定义在另一个类中的类.为什么需要使用内部类呢?其主要原因有以下三点: 1.内部类方法可以访问该类定义所在的作用域中的数据,包括私有的数据. 2.内部类可以对同一个包中 ...

  5. hdu 2577 模拟

    #include<stdio.h> #define N 200 char s[N]; int judgeup(char c) { if(c>='A'&&c<=' ...

  6. osEye.Net:离别是为了将来的重逢

    这一时刻已经成为osEye历史..... 在热心网友的关怀和鼓励之下,osEye.net将继续运行着,感谢你们陪伴osEye一起走过.... 与你相知相恋已经有4个年头了,你的成长到成熟都让我历历在目 ...

  7. 使用CSStickyHeaderFlowLayout实现头部固定的CollectionView

    近期流行的一种界面效果,是瀑布流的header固定,也叫sticky header或者parallax.对于UITableView,能够比較方便地让table header固定,可是对于UIColle ...

  8. 【翻译自mos文章】 在错误的从os级别remove掉 trace file 之后,怎么找到该trace file的内容?

    在错误的从os级别remove掉 trace file 之后,怎么找到该trace file的内容? 參考原文: How to Find the Content of Trace File Gener ...

  9. muduo总结

    总结说的有的过大,算是对自己学习的一个总结.兴许会不断补充. 模型总结 muduo是基于非堵塞的IO和事件驱动的网络库. muduo的总体结构时one loop per thread+threadpo ...

  10. jQuery的父,子,兄弟节点查找方法

    以下罗列一下jQery下节点查找的方法: jQuery.parent(expr) 找父亲节点.能够传入expr进行过滤,比方$("span").parent()或者$(" ...