一道优雅面试题分析js中fn()和return fn()的区别


Posted in Javascript onJuly 05, 2016

在js中,经常会遇到在函数里调用其它函数的情况,这时候会有 fn() 这种调用方式,还有一种是 return fn() 这种调用方式,一些初学者经常会被这两种方式给绕晕了。这里用一个优雅的面试题来分析一下两种方式的不同之处。 

var i = 0;
function fn(){
 i++;
 if(i < 10){
 fn();
 }else{
 return i;
 }
}

var result = fn();
console.log(result);

这是一道隐藏了坑的面试题,看似很简单,大部分人可能想都不想就答出了10。而实际上运行可知打印出来的是 undefined。这道陷阱题很直观的体现出了前面所说的问题,当我们将执行fn的那一行修改为: 

var i = 0;
function fn(){
 i++;
 if(i < 10){
 return fn();
 }else{
 return i;
 }
}

var result = fn();
console.log(result);

这时,会发现打印出来的结果终于不负众望的是 10 了。

为什么这里加不加return区别会这么大? 

这里的主要原因很简单,JavaScript的函数都是有默认返回值的,如果函数结尾不写return,会默认返回undefined,这就是为什么在chrome的console控制台里,写代码经常下面会出现一行undefined的原因。 

再仔细看看这个例子,当i自增到9的时候,也就是倒数第二次递归调用fn的那一次,如果没有return,这一次执行完fn,会默认return undefined,而不会继续下一次递归了。当加上了 return,在这里则会继续最后一次递归,即i=10的时候,跳入else里面返回得到正确的10。 

说到这里,可以引申出一个更为经典的例子,著名的二分查找法: 

var mid = Math.floor((arr.length - 1) / 2);

function search(n, mid) {
 if (n > arr[mid]) {
 mid = Math.floor((mid + arr.length) / 2);
 return search(n, mid);
 } else if (n < arr[mid]) {
 mid = Math.floor((mid - 1) / 2);
 return search(n, mid);
 } else {
 return mid;
 }
}

var index = search(n, mid);
console.log(index);

二分查找法也是需要多次递归调用,很多新手在第一次实现这个算法的时候经常会犯的一个错误就是忘记在递归的函数前加上return,最后导致返回结果是undefined,这里的道理也和前面是类似的,不加return,会导致递归后,直接返回undefined,不会继续下一次递归。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Javascript 相关文章推荐
在一个form用一个SUBMIT(或button)分别提交到两个处理表单页面的代码
Feb 15 Javascript
服务端 VBScript 与 JScript 几个相同特性的写法 By shawl.qiu
Mar 06 Javascript
jQuery.extend 函数详解
Feb 03 Javascript
给页面渲染时间加速 干掉Dom Level 0 Event
Dec 19 Javascript
js实现目录定位正文示例
Nov 14 Javascript
浅析JavaScript作用域链、执行上下文与闭包
Feb 01 Javascript
纯JS代码实现气泡效果
May 04 Javascript
js判断checkbox是否选中个数的方法(超简单)
Aug 19 Javascript
vue-cli项目配置多环境的详细操作过程
Oct 30 Javascript
在Vant的基础上实现添加表单验证框架的方法示例
Dec 05 Javascript
Vue 列表上下过渡效果的实例代码
Jun 25 Javascript
VUE.CLI4.0配置多页面入口的实现
Nov 25 Javascript
JS实现环形进度条(从0到100%)效果
Jul 05 #Javascript
JQuery组件基于Bootstrap的DropDownList(完整版)
Jul 05 #Javascript
结合代码图文讲解JavaScript中的作用域与作用域链
Jul 05 #Javascript
jQuery的Each比JS原生for循环性能慢很多的原因
Jul 05 #Javascript
Node.js实现文件上传
Jul 05 #Javascript
JavaScript数组方法大全(推荐)
Jul 05 #Javascript
JS与HTML结合使用marquee标签实现无缝滚动效果代码
Jul 05 #Javascript
You might like
PHP4 与 MySQL 数据库操作函数详解
2006/10/09 PHP
PHP中对用户身份认证实现两种方法
2011/06/04 PHP
php之static静态属性与静态方法实例分析
2015/07/30 PHP
关于扩展 Laravel 默认 Session 中间件导致的 Session 写入失效问题分析
2016/01/08 PHP
PHP实现的折半查询算法示例
2017/10/09 PHP
Javascript在IE或Firefox下获取鼠标位置的代码
2009/12/18 Javascript
IE和Firefox下event事件杂谈
2009/12/18 Javascript
js如何判断不同系统的浏览器类型
2013/10/28 Javascript
jQuery+css3动画属性制作猎豹浏览器宽屏banner焦点图
2015/03/16 Javascript
Grunt入门教程(自动任务运行器)
2015/08/06 Javascript
jQuery+ajax实现滚动到页面底部自动加载图文列表效果(类似图片懒加载)
2016/06/07 Javascript
Form表单按回车自动提交表单的实现方法
2016/11/18 Javascript
原生js实现可拖拽效果
2017/02/28 Javascript
js学习总结之DOM2兼容处理重复问题的解决方法
2017/07/27 Javascript
vue中for循环更改数据的实例代码(数据变化但页面数据未变)
2017/09/15 Javascript
vue动态路由:路由参数改变,视图不更新问题的解决
2019/11/05 Javascript
axios解决高并发的方法:axios.all()与axios.spread()的操作
2020/11/09 Javascript
[02:03]DOTA2亚洲邀请赛 HGT战队出场宣传片
2015/02/07 DOTA
[00:36]DOTA2勇士令状莱恩声望物品——冥晶之厄展示
2018/05/25 DOTA
利用Python实现图书超期提醒
2016/08/02 Python
详解MySQL数据类型int(M)中M的含义
2016/11/20 Python
python实现搜索文本文件内容脚本
2018/06/22 Python
Pytorch 保存模型生成图片方式
2020/01/10 Python
Django 返回json数据的实现示例
2020/03/05 Python
python实现感知机模型的示例
2020/09/30 Python
联想美国官方商城:Lenovo美国
2017/06/19 全球购物
Charles & Colvard官网:美国莫桑石品牌
2019/06/05 全球购物
华为消费者德国官方网站:HUAWEI德国
2020/11/03 全球购物
屈臣氏泰国官网:Watsons TH
2021/02/23 全球购物
材料加工硕士生求职信
2013/10/10 职场文书
道路建设实施方案
2014/03/18 职场文书
保密协议书范本
2014/04/22 职场文书
大学国际贸易专业自荐信
2014/06/05 职场文书
宾馆安全管理制度
2015/08/06 职场文书
css3 filter属性的使用简介
2021/03/31 HTML / CSS
使用Python获取字典键对应值的方法
2022/04/26 Python