递归

上一篇blog里,用js实现了quicksort算法,quicksort算法是一个递归调用过程。

递归是一种非常强大的编程思想,它广泛的的存在于各种语言当中,尤其lisp的各种方言中,大量的使用递归来实现循环操作。

还有一本小书叫《The litter schemer》,就是让编程的人递归的思考问题(thinking recursively)。

本篇呈接上篇,是对递归思想的实践——在学习常见的算法的同时,运用递归思想。

理解了递归思想,以前觉得挺难的算法也觉得不是想象的那么难了。

mergsort in javascript

  1. 最简单的情况是对两个元素的数组排序;
  2. 如果两个数组已经排序好了,那么,再将这两个数组合并为一个有序的数组是比较容易的;
  3. 一个混乱的数组,总能将其分成两部分,再将分成的两部分再分为两部分,直到有一个部分只包含两个元素,那么就回到第1步的情况,对两个元素的数组进行排序;
  4. 这是一个递归过程
function first(l){
return l[0];
}
function rest(l){
return l.slice(1);
} /** 实现2
* l1 l2 是已经从小到大排序好的数组
* 返回一个l1 l2合并后的,从小到大排序好的数组
* */
function mergelist(l1,l2){
var ret
if(l1.length == 0){
ret = l2;
}else if(l2.length == 0){
ret = l1;
}else if(first(l1)>first(l2)){//将小的放前面
ret = [].concat(first(l2),
mergelist(l1,rest(l2)))
}else{
ret = [].concat(first(l1),//将小的元素放前面
mergelist(l2,rest(l1)))
}
return ret;
} // console.log(mergelist([2,3],[1,4,5]));
// console.log(mergelist([1,4,5],[2,3]));
// --> [ 1, 2, 3, 4, 5 ] // 下面这个操作是对一组数数组排序的最小单位排序操作——因为
// 一个包含任意元素的数组的排序最终都能逐渐的递归细分为比较两个元素的大小
// 切分操作由下面的divide函数来完成
// console.log(mergelist([5],[2]));
// --> [ 2, 5 ] // 实现3中的拆分
// 将一个数组分为两部分
function divide(l){
var len = l.length
, mid = Math.floor(len/2);
return [l.slice(0,mid),l.slice(mid,len)]
} // console.log(divide([1,4,5,2,3]));
//--> [ [ 1, 4 ], [ 5, 2, 3 ] ] // 整体实现
function mergesort(l1){
var ab = divide(l1)
, a = ab[0]
, b = ab[1]; if(a.length == 0){
return b;
}else if(b.length == 0){
return a;
}else if(a.length == 1){
return [].concat(mergelist(a,mergesort(b)));
}else if(b.length == 1){
return [].concat(mergelist(b,mergesort(a)));
}else{
return [].concat(mergelist(mergesort(a),
mergesort(b)));
}
} console.log(mergesort([5,2,3,4]));
console.log(mergesort([5,3,4]));
// console.log(mergesort([1,4,5,2,3])); // 注意[].concat方法的运用
// [].concat(1,[2,3]) -> [1,2,3]
// [].concat([1],[2,3]) -> [1,2,3]
// [].concat(1,2,3) -> [1,2,3]

merge sort 的javascript实现的更多相关文章

  1. [Algorithms] Divide and Recurse Over an Array with Merge Sort in JavaScript

    Merge sort is a recursive sorting algorithm. If you don't understand recursion, I recommend finding ...

  2. Write a merge sort program

    Merge Sort- Recursion Write a merge sort program in JavaScript. Sample array : [34, 7, 23, 32, 5, 62 ...

  3. 【算法】归并排序(Merge Sort)(五)

    归并排序(Merge Sort) 归并排序是建立在归并操作上的一种有效的排序算法.该算法是采用分治法(Divide and Conquer)的一个非常典型的应用.将已有序的子序列合并,得到完全有序的序 ...

  4. [算法]——归并排序(Merge Sort)

    归并排序(Merge Sort)与快速排序思想类似:将待排序数据分成两部分,继续将两个子部分进行递归的归并排序:然后将已经有序的两个子部分进行合并,最终完成排序.其时间复杂度与快速排序均为O(nlog ...

  5. SQL Tuning 基础概述06 - 表的关联方式:Nested Loops Join,Merge Sort Join & Hash Join

    nested loops join(嵌套循环)   驱动表返回几条结果集,被驱动表访问多少次,有驱动顺序,无须排序,无任何限制. 驱动表限制条件有索引,被驱动表连接条件有索引. hints:use_n ...

  6. 归并排序(Merge Sort)

    归并排序是建立在归并操作上的一种有效的排序算法,该算法是采用分治法(Divide and Conquer)的一个非常典型的应用.将已有序的子序列合并,得到完全有序的序列:即先使每个子序列有序,再使子序 ...

  7. 归并排序(merge sort)

    M erge sort is based on the divide-and-conquer paradigm. Its worst-case running time has a lower ord ...

  8. Summary: Merge Sort of Array && 求逆序对

    常用算法(后面有inplace版本): package ArrayMergeSort; import java.util.Arrays; public class Solution { public ...

  9. 基础排序算法之并归排序(Merge Sort)

    并归排序是学习分治法 (Merge Sort) 的好例子.而且它相对于选择,插入,冒泡排序来说,算法性能有一定提升.我首先会描述要解决的问题,并给出一个并归排序的例子.之后是算法的思路以及给出伪代码. ...

随机推荐

  1. 1021 docker初识

    docker与虚拟机相比,没有虚拟化内核,转而使用宿主机的内核.因此docker更轻更快 docker缺点:后端兼容性测试需求.把软件安装在不同的操作系统上进行测试,观察软件运行是否良好. 不能用do ...

  2. 通过 Lua 扩展 NGINX 实现的可伸缩的 Web 平台OpenResty®

    OpenResty® 是一个基于 Nginx 与 Lua 的高性能 Web 平台,其内部集成了大量精良的 Lua 库.第三方模块以及大多数的依赖项.用于方便地搭建能够处理超高并发.扩展性极高的动态 W ...

  3. apache启动不了, 查找错误

    今天apache启动不了, 本来以为是端口冲突, 用 cmd-> netstat -aon|findstr "80"  或 tasklist|findstr "80 ...

  4. Visual Basic 函数速查

    Calendar 常数 可在代码中的任何地方用下列常数代替实际值: 常数 值 描述 vbCalGreg 0 指出使用的是阳历. vbCalHijri 1 指出使用的是伊斯兰历法. Color 常数 可 ...

  5. JVM 调优参数设置

    先看Linux内存大小(假设为2G) cat /proc/meminfo |grep MemTotal 查看java初始配置 java -XX:+PrintFlagsInitial Tomcat配置 ...

  6. 105. Construct Binary Tree from Preorder and Inorder Traversal (Tree; DFS)

    Given preorder and inorder traversal of a tree, construct the binary tree. Note: You may assume that ...

  7. JS如何判断浏览器类型,如何模拟浏览器类型(模拟微信浏览器)

    一.前言 在编写前端代码时,为了页面兼容性,我们往往需要考虑不同的浏览器类型 而这就需要在前端代码中进行识别和区分 接下来就来谈谈对浏览器类型的识别 二.正文 (一).查看浏览器类型的核心代码 var ...

  8. shader一般都是用工具调试的

    N卡的话用nvidia的nVidia FX Composer, A卡的话用ATI的render monkey 顶点着色器从何方拿到这些数据?在U3D环境下,答案是从绑定到game object中的Me ...

  9. C++中public、protected以及private的使用

    相比C语言,C++中通过class/struct来定义既包含数据,又包含行为的结构,从而支持了“对象”.现实世界中,一个人(一个对象)通常 拥有一些资产(数据),并且掌握某些技能(行为),并且这些资产 ...

  10. ROS 消息发布器和订阅器Publisher, Subscriber

    博客参考:https://www.2cto.com/kf/201705/639776.html 1.编写发布器节点节点(Node) 是指 ROS 网络中可执行文件.接下来,将会创建一个发布器节点(“t ...