(转载)JavaScript中匿名函数,函数直接量和闭包


Posted in Javascript onMay 08, 2007

原文出处: http://www.dnew.cn/post/196.htm

先看下下面几种写法

1.function f(x){return x*x;};f(x);

2.(function(x){return x*x;})(x);

3.(function(x){return x*x;}(x));

第一种我们应该都很熟悉了,这是我们经常使用的写法。第二第三种都是匿名函数的写法。

--------------------------------------------------------------------------------

第二种
可以这样理解:

var f=function(x) {return x*x;};f()

那我们不通过f这个变量来引用函数就是

function(){}()

然而这样肯定是错误的就像

var f=1+2;
f=f*0;

var f=1+2*0;

结果不同一样。
要得到正确结果只能:

f=(1+2)*0;

也就是要明确的标识出程序块,即:

(function(){})()

肯你有疑问:括号“()”到底是不是起到了标识代码块的作用?
我们可以用JavaScript的内置函数检测一下!
举一个最简单的例子:

alert(4)

这段代码会弹出提示内容是“4”
改成这样
(alert)(4)

可以看到执行的效果和上一段代码一样。

这种形式的函数执行也被很多JavaScript框架所采用。

--------------------------------------------------------------------------------

第三种,如果你用过jsvm框架的话就会发现里面的代码使用了这种形式。
那如何解释第三种情况呢?
为了弄明白浏览器是如何理解这样的写法的,我们可以利用一下Mozilla Firefox的错误控制台功能。
在代码中插入一段错误代码,代码段如下:

(function(s){s+s}(1)).splice();

打开Mozilla Firefox的错误控制台,可以看到有如下的错误提示

错误: (function (s) {})(1) has no properties
源文件:file:///C:/Documents…….html
行:18

可以认为,浏览器对于
(function(s){s+s}(1))
这样的代码按照

(function (s) {s+s})(1)
来解析的。

--------------------------------------------------------------------------------

到此可能你有这样的认识:

function f(x){return x*x;};f(x);==(function(x){return x*x;})(x);==(function(x){return x*x;}(x));

但是他们还是有区别的,
首先,对于像第二和第三种形式,其它的函数和代码是不可能调用所定义的函数的,有一种说发把这样的函数称为匿名函数或者函数直接量。
其次,第二和第三种形式执行的函数,中间变量不会污染到全局命名空间,你可以把中间的代码看作纯粹的子过程调用。
当然使用后面两种形式的函数定义可以很容易的实现闭包。
看一个例子:

/*
http://jibbering.com/faq/faq_notes/closures.html(Dnew.CN注)
A global variable - getImgInPositionedDivHtml - is declared and
  assigned the value of an inner function expression returned from
  a one-time call to an outer function expression.

  That inner function returns a string of HTML that represents an
  absolutely positioned DIV wrapped round an IMG element, such that
  all of the variable attribute values are provided as parameters
  to the function call:-
*/
var getImgInPositionedDivHtml = (function(){
   /* The - buffAr - Array is assigned to a local variable of the
      outer function expression. It is only created once and that one
      instance of the array is available to the inner function so that
      it can be used on each execution of that inner function.

      Empty strings are used as placeholders for the date that is to
      be inserted into the Array by the inner function:-
   */
   var buffAr = [
       '<div id="',
       '',   //index 1, DIV ID attribute
       '" style="position:absolute;top:',
       '',   //index 3, DIV top position
       'px;left:',
       '',   //index 5, DIV left position
       'px;width:',
       '',   //index 7, DIV width
       'px;height:',
       '',   //index 9, DIV height
       'px;overflow:hidden;\"><img src=\"',
       '',   //index 11, IMG URL
       '\" width=\"',
       '',   //index 13, IMG width
       '\" height=\"',
       '',   //index 15, IMG height
       '\" alt=\"',
       '',   //index 17, IMG alt text
       '\"><\/div>'
   ];
   /* Return the inner function object that is the result of the
      evaluation of a function expression. It is this inner function
      object that will be executed on each call to -
      getImgInPositionedDivHtml( ... ) -:-
   */
   return (function(url, id, width, height, top, left, altText){
       /* Assign the various parameters to the corresponding
          locations in the buffer array:-
       */
       buffAr[1] = id;
       buffAr[3] = top;
       buffAr[5] = left;
       buffAr[13] = (buffAr[7] = width);
       buffAr[15] = (buffAr[9] = height);
       buffAr[11] = url;
       buffAr[17] = altText;
       /* Return the string created by joining each element in the
          array using an empty string (which is the same as just
          joining the elements together):-
       */
       return buffAr.join('');
   }); //:End of inner function expression.
})();
/*^^- :The inline execution of the outer function expression. */

Javascript 相关文章推荐
Javascript学习笔记一 之 数据类型
Dec 15 Javascript
document.write()及其输出内容的样式、位置控制
Aug 12 Javascript
JavaScript如何自定义trim方法
Jul 28 Javascript
js接收并转化Java中的数组对象的方法
Aug 11 Javascript
jQuery实现文章图片弹出放大效果
Apr 06 jQuery
jquery请求servlet实现ajax异步请求的示例
Jun 03 jQuery
JS设置手机验证码60s等待实现代码
Jun 14 Javascript
vue.js如何将echarts封装为组件一键使用详解
Oct 10 Javascript
jQuery除指定区域外点击任何地方隐藏DIV功能
Nov 13 jQuery
详解如何在nuxt中添加proxyTable代理
Aug 10 Javascript
微信小程序webview组件交互,内联h5页面并网页实现微信支付实现解析
Aug 16 Javascript
vue 解决异步数据更新问题
Oct 29 Javascript
阻止JavaScript事件冒泡传递(cancelBubble 、stopPropagation)
May 08 #Javascript
延时重复执行函数 lLoopRun.js
May 08 #Javascript
用js判断浏览器是否是IE的比较好的办法
May 08 #Javascript
[全兼容哦]--实用、简洁、炫酷的页面转入效果loing
May 07 #Javascript
javascript之水平横向滚动歌词同步的应用
May 07 #Javascript
javascript之ESC(第二类混淆)
May 06 #Javascript
通过Unicode转义序列来加密,按你说的可以算是混淆吧
May 06 #Javascript
You might like
定义php常量的详解
2013/06/09 PHP
php中自定义函数dump查看数组信息类似var_dump
2014/01/27 PHP
PHP中trait使用方法详细介绍
2017/05/21 PHP
php7性能提升的原因详解
2019/10/13 PHP
jQuery学习笔记之Helloworld
2010/12/22 Javascript
jQuery操作select的实例代码
2012/06/14 Javascript
jQuery操作input type=radio的实现代码
2012/06/14 Javascript
jquery得到font-size属性值实现代码
2013/09/30 Javascript
文本框只能选择数据到文本框禁止手动输入
2013/11/22 Javascript
jQuery之过滤元素操作小结
2013/11/30 Javascript
Javascript学习笔记之 函数篇(二) : this 的工作机制
2014/06/24 Javascript
js用拖动滑块来控制图片大小的方法
2015/02/27 Javascript
JavaScript中的Repaint和Reflow用法详解
2015/07/27 Javascript
javascript精确统计网站访问量实例代码
2015/12/19 Javascript
angular2中Http请求原理与用法详解
2018/01/11 Javascript
vue+VeeValidate 校验范围实例详解(部分校验,全部校验)
2018/10/19 Javascript
详解基于electron制作一个node压缩图片的桌面应用
2019/01/29 Javascript
react使用CSS实现react动画功能示例
2020/05/18 Javascript
9个JavaScript日常开发小技巧
2020/10/06 Javascript
[33:42]LGD vs OG 2018国际邀请赛小组赛BO2 第一场 8.16
2018/08/17 DOTA
关于python的list相关知识(推荐)
2017/08/30 Python
全面了解Nginx, WSGI, Flask之间的关系
2018/01/09 Python
Python 获取ftp服务器文件时间的方法
2019/07/02 Python
Python内置加密模块用法解析
2019/11/25 Python
Python如何将模块打包并发布
2020/08/30 Python
Python HTMLTestRunner如何下载生成报告
2020/09/04 Python
CSS3 实现雷达扫描图的示例代码
2020/09/21 HTML / CSS
Fanatics法国官网:美国体育电商
2019/08/27 全球购物
军校制空专业毕业生自我鉴定
2013/11/16 职场文书
物业消防安全责任书
2014/07/23 职场文书
企业委托书范本
2014/09/13 职场文书
2014年银行员工年终自我评价
2014/09/19 职场文书
工资收入证明
2014/10/07 职场文书
机关单位工作失职检讨书
2014/11/20 职场文书
体检通知范文
2015/04/21 职场文书
2019自荐信该如何写呢?
2019/07/05 职场文书