(转载)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 相关文章推荐
拖动一个HTML元素
Dec 22 Javascript
JQuery自定义事件的应用 JQuery最佳实践
Aug 01 Javascript
多个js与css文件的合并方法详细说明
Dec 26 Javascript
jquery自定义属性(类型/属性值)
May 21 Javascript
深入理解Javascript作用域与变量提升
Dec 09 Javascript
详解Vue.js组件可复用性的混合(mixin)方式和自定义指令
Sep 06 Javascript
React Native自定义控件底部抽屉菜单的示例
Feb 08 Javascript
Koa 中的错误处理解析
Apr 09 Javascript
Js实现复选框的全选、全不选反选功能代码实例
Feb 28 Javascript
JavaScript中CreateTextFile函数
Aug 30 Javascript
vue动态设置路由权限的主要思路
Jan 13 Vue.js
javascript实现用户必须勾选协议实例讲解
Mar 24 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
PHP4实际应用经验篇(1)
2006/10/09 PHP
php中将图片gif,jpg或mysql longblob或blob字段值转换成16进制字符串
2011/08/23 PHP
php实现水仙花数示例分享
2014/04/03 PHP
php编程中echo用逗号和用点号连接的区别
2016/03/26 PHP
Javascript中查找不以XX字符结尾的单词示例代码
2013/10/15 Javascript
jQuery unbind 删除绑定事件详解
2016/05/24 Javascript
javascript self对象使用详解
2016/10/18 Javascript
Vue指令的钩子函数使用方法
2017/03/20 Javascript
JS验证input输入框(字母,数字,符号,中文)
2017/03/23 Javascript
JavaScript常用8种数组去重代码实例
2020/09/09 Javascript
[03:05]DOTA2英雄基础教程 嗜血狂魔
2013/12/10 DOTA
Python常用的内置序列结构(列表、元组、字典)学习笔记
2016/07/08 Python
python numpy函数中的linspace创建等差数列详解
2017/10/13 Python
使用pandas将numpy中的数组数据保存到csv文件的方法
2018/06/14 Python
Python3 导入上级目录中的模块实例
2019/02/16 Python
Django shell调试models输出的SQL语句方法
2019/08/29 Python
基于python中__add__函数的用法
2019/11/25 Python
python 实现turtle画图并导出图片格式的文件
2019/12/07 Python
基于matplotlib中ion()和ioff()的使用详解
2020/06/16 Python
Selenium及python实现滚动操作多种方法
2020/07/21 Python
Python命名空间及作用域原理实例解析
2020/08/12 Python
文件中有一组整数,要求排序后输出到另一个文件中
2012/01/04 面试题
SQL注入攻击的种类有哪些
2013/12/30 面试题
介绍一下Java的事务处理
2012/12/07 面试题
浙江文明网签名寄语
2014/01/18 职场文书
创建文明学校实施方案
2014/03/11 职场文书
护士优质服务演讲稿
2014/08/26 职场文书
2014年个人委托书范本
2014/10/13 职场文书
实习计划书范文
2015/01/16 职场文书
社会实践单位意见
2015/06/05 职场文书
公司保密管理制度
2015/08/04 职场文书
2019年怎样写好导游词?
2019/07/02 职场文书
送给客户微信问候语!
2019/07/04 职场文书
导游词之河姆渡遗址博物馆
2019/10/10 职场文书
mysql使用 not int 子查询隐含陷阱
2022/04/12 MySQL
nginx容器方式反向代理实战
2022/04/18 Servers