js fuction函数内return一个内部函数详解

本文涉及的产品
日志服务 SLS,月写入数据量 50GB 1个月
简介: 今天在网上,看到一篇关于js函数难点的文章,js函数的一些难点。在那上面提了一下,关于js函数返回另一个函数的问题,并附上了一道面试题:[javascript] view plain copy var add = function(x){      var sum = 1;  ...

今天在网上,看到一篇关于js函数难点的文章,js函数的一些难点。在那上面提了一下,关于js函数返回另一个函数的问题,并附上了一道面试题:

[javascript] view plain copy 在CODE上查看代码片派生到我的代码片
  1. var add = function(x){  
  2.     var sum = 1;  
  3.     var tmp = function(x){  
  4.         sum = sum + x;  
  5.     return tmp;  
  6.     }  
  7.     tmp.toString = function(){  
  8.         return sum;  
  9.     }  
  10.     return tmp;  
  11. // alert(add(1)(2)(3)) --> 6  

接下来,就来详细的解读返回另一个函数的问题。

其实我是从Java转过来的,一开始看到那篇文章,我对于返回另一个函数并没有什么认识,我之所以写这篇文章是因为,在那里面有一点让我感到奇怪,那就是最后的调用方式

[javascript] view plain copy 在CODE上查看代码片派生到我的代码片
  1. add(1)(2)(3)  

由于在java中,我没有见到过这样的函数调用方式,所以引起了我的注意,我决定去研究研究;下面就将我的研究分享出来,当然如果你对此已经有了深刻的认识,你可以选择跳过,或者对于不足的地方,给出指点微笑。好了闲话不多说,进入正题。

我们来看一个最简单的例子:

[javascript] view plain copy 在CODE上查看代码片派生到我的代码片
  1. function create1(pro) {  
  2.     console.log("pro : " + pro);  
  3.     return function(obj1, obj2){  
  4.         console.log(obj1 + " -- " + obj2);  
  5.         return obj1 + obj2;  
  6.     }  
  7. }  

我构建了一个简单的函数create1,并且有一个返回值,返回值是一个内部函数。函数构建完了,接下来进行调用:

[javascript] view plain copy 在CODE上查看代码片派生到我的代码片
  1. var c1 = create1("pro"); // 创建函数  

如果按照我之前的理解,当我调用了这个方法后,应该会打印出 pro : pro,接着然后报错的。如果你看完过后,也跟我有一样的想法,那恭喜你想多了或者有了固型思维微笑

。真实的是当我们通过上面的代码调用的时候,日志是打印出了  pro : pro ,但是并没有报错,并且我们反复来回的调用过后,也只是来回的打印相同的日志。这也就说明这个时候,只是进入了create1()方法,并没有进入到该函数的内部函数内。通过面试题的启发,我在试着调用了一次,发现打印出了后续的。

[java] view plain copy 在CODE上查看代码片派生到我的代码片
  1. c1(1, 2); // 调用函数  

这样就打印出了下面的日志;这说明其实我们一开始调用方法的时候,其实是并没有进入到里层的函数的,只是进入了外层函数体,我们只有再调用才能进入里层函数体,并且这个时候,我们重复上面的调用,他只会是调用里层的函数体,并没有外面的函数体。

类似这种函数返回另一个函数的,我们第一次调用只是构建了一个外层函数体对象,只有有后续的调用,才能调用内层函数体,并且重复调用,只会重复内层函数体。
不要急,还没有完,后面还有……

接下来,我们看一看另一种情况,我们先声明一个函数,用来做加法运算:

[javascript] view plain copy 在CODE上查看代码片派生到我的代码片
  1. function infun(obj1, obj2) {  
  2.     console.log(obj1 + " -- " + obj2);  
  3.         return obj1 + obj2;  
  4. }  

然后再声明一个函数,在该函数中调用上面声明的函数:

[javascript] view plain copy 在CODE上查看代码片派生到我的代码片
  1. function create2(pro) {  
  2.     console.log("pro = " + pro);  
  3.         return infun(obj1, obj2); // 这个时候,会报错  
  4. }  

最后是调用:

[javascript] view plain copy 在CODE上查看代码片派生到我的代码片
  1. var c1 = create2("pro");   

查看日志:

pro = pro
‌Uncaught ReferenceError: obj1 is not defined

会发现,打印出了一条日志后,接着抛出了异常。对方法做一下改动,

[javascript] view plain copy 在CODE上查看代码片派生到我的代码片
  1. function create2(pro) {  
  2.     console.log("pro = " + pro);  
  3.     var obj1 = 1, obj2 = 2;  
  4.     return infun(obj1, obj2); // 这个时候,会报错  
  5. }  

在调用会发现正常运行,并且打印出了两条日志记录。

这说明,类似于这种,在一个函数内返回一个已经声明的函数,其实是调用已经声明的函数,跟上面的情况是不一样的。
好了,现在回过头来,仔细看看开头的面试题,就会发现一切都明了了:

[javascript] view plain copy 在CODE上查看代码片派生到我的代码片
  1. // 声明一个函数表达式  
  2. var add = function(x){  
  3.     var sum = 1;  
  4.     // 在函数表达式内部有一个求和的内部函数  
  5.     var tmp = function(x){  
  6.         sum = sum + x;// 求和  
  7.         return tmp;  
  8.     }  
  9.     // 构建一个函数体的toString()函数  
  10.     tmp.toString = function(){  
  11.         return sum;  
  12.     }  
  13.     return tmp; // 返回的是一个函数体,如果该函数体有toString()方法,则会调用函数体的toString()方法  
  14. }  

然后再来看看调用:

[javascript] view plain copy 在CODE上查看代码片派生到我的代码片
  1. alert(add(1)(2)(3))  

结果为6,至于原因就跟我们第一种讨论的情况一样,接下来,我们反复调用:

[javascript] view plain copy 在CODE上查看代码片派生到我的代码片
  1. // 以下结果输出为:6  
  2. alert(add(10)(2)(3))  
  3. alert(add(100)(2)(3))  
  4. // 下面的结果输出变了  
  5. alert(add(1)(3)(3))  
  6. alert(add(1)(2)(5))  


相关实践学习
日志服务之使用Nginx模式采集日志
本文介绍如何通过日志服务控制台创建Nginx模式的Logtail配置快速采集Nginx日志并进行多维度分析。
目录
相关文章
|
1月前
|
JavaScript 前端开发 Java
[JS]同事:这次就算了,下班回去赶紧补补内置函数,再犯肯定被主管骂
本文介绍了JavaScript中常用的函数和方法,包括通用函数、Global对象函数以及数组相关函数。详细列出了每个函数的参数、返回值及使用说明,并提供了示例代码。文章强调了函数的学习应结合源码和实践,适合JavaScript初学者和进阶开发者参考。
41 2
[JS]同事:这次就算了,下班回去赶紧补补内置函数,再犯肯定被主管骂
|
1月前
|
前端开发 JavaScript 开发者
除了 Generator 函数,还有哪些 JavaScript 异步编程解决方案?
【10月更文挑战第30天】开发者可以根据具体的项目情况选择合适的方式来处理异步操作,以实现高效、可读和易于维护的代码。
|
2月前
|
JavaScript 前端开发
JavaScript 函数语法
JavaScript 函数是使用 `function` 关键词定义的代码块,可在调用时执行特定任务。函数可以无参或带参,参数用于传递值并在函数内部使用。函数调用可在事件触发时进行,如用户点击按钮。JavaScript 对大小写敏感,函数名和关键词必须严格匹配。示例中展示了如何通过不同参数调用函数以生成不同的输出。
|
2月前
|
存储 JavaScript 前端开发
JS函数提升 变量提升
【10月更文挑战第6天】函数提升和变量提升是 JavaScript 语言的重要特性,但它们也可能带来一些困惑和潜在的问题。通过深入理解和掌握它们的原理和表现,开发者可以更好地编写和维护 JavaScript 代码,避免因不了解这些机制而导致的错误和不一致。同时,不断提高对执行上下文等相关概念的认识,将有助于提升对 JavaScript 语言的整体理解和运用能力。
|
2月前
|
JavaScript 前端开发
js教程——函数
js教程——函数
42 4
|
2月前
|
存储 JavaScript 前端开发
js中函数、方法、对象的区别
js中函数、方法、对象的区别
21 2
|
2月前
|
JavaScript 前端开发 Java
【javaScript数组,函数】的基础知识点
【javaScript数组,函数】的基础知识点
27 5
|
2月前
|
JavaScript 前端开发
Node.js 函数
10月更文挑战第5天
24 3
|
2月前
|
前端开发 JavaScript
探索JavaScript函数基础
探索JavaScript函数基础
19 3
|
2月前
|
存储 JavaScript 前端开发
JavaScript数据类型全解:编写通用函数,精准判断各种数据类型
JavaScript数据类型全解:编写通用函数,精准判断各种数据类型
26 0