(转载)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 相关文章推荐
angularjs客户端实现压缩图片文件并上传实例
Jul 06 Javascript
js输出数据精确到小数点后n位代码
Jul 02 Javascript
BootStrap Typeahead自动补全插件实例代码
Aug 10 Javascript
js select实现省市区联动选择
Apr 17 Javascript
关于iframe跨域POST提交的方法示例
Jan 15 Javascript
react实现pure render时bind(this)隐患需注意!
Mar 09 Javascript
JavaScript实现三级联动效果
Jul 15 Javascript
Bootstrap Table快速完美搭建后台管理系统
Sep 20 Javascript
JS动态添加的div点击跳转到另一页面实现代码
Sep 30 Javascript
Vue+webpack+Element 兼容问题总结(小结)
Aug 16 Javascript
JS实现电话号码的字母组合算法示例
Feb 26 Javascript
layer.js open 隐藏滚动条的例子
Sep 05 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中文本操作的类
2007/03/17 PHP
php实现两个数组相加的方法
2015/02/17 PHP
php编程中echo用逗号和用点号连接的区别
2016/03/26 PHP
PHP 应用容器化以及部署方法
2018/02/12 PHP
几个高效,简洁的字符处理函数
2007/04/12 Javascript
jquery 弹出登录窗口实现代码
2009/12/24 Javascript
JS的Document属性和方法小结
2013/09/17 Javascript
javascript阻止浏览器后退事件防止误操作清空表单
2013/11/22 Javascript
jQuery插件Timelinr 实现时间轴特效
2015/10/04 Javascript
JavaScript Split()方法
2015/12/18 Javascript
基于JS实现回到页面顶部的五种写法(从实现到增强)
2016/09/03 Javascript
Dropzone.js实现文件拖拽上传功能(附源码下载)
2016/11/22 Javascript
js实现旋转木马效果
2017/03/17 Javascript
在微信小程序里使用watch和computed的方法
2018/08/02 Javascript
编写Python脚本批量下载DesktopNexus壁纸的教程
2015/05/06 Python
Python编程pygame模块实现移动的小车示例代码
2018/01/03 Python
Python设置在shell脚本中自动补全功能的方法
2018/06/25 Python
Python3使用pandas模块读写excel操作示例
2018/07/03 Python
python tkinter实现界面切换的示例代码
2019/06/14 Python
利用Python实现Shp格式向GeoJSON的转换方法
2019/07/09 Python
Python协程操作之gevent(yield阻塞,greenlet),协程实现多任务(有规律的交替协作执行)用法详解
2019/10/14 Python
Python 生成VOC格式的标签实例
2020/03/10 Python
Selenium元素定位的30种方式(史上最全)
2020/05/11 Python
django 将自带的数据库sqlite3改成mysql实例
2020/07/09 Python
HTML5自定义视频播放器源码
2020/01/06 HTML / CSS
英国领先的男士美容护发用品公司:Mankind
2016/08/31 全球购物
医院护理人员的自我评价分享
2013/10/04 职场文书
英文简历中的自荐信范文
2013/12/14 职场文书
安全生产活动月方案
2014/03/09 职场文书
调解协议书
2014/04/16 职场文书
搞笑爱情保证书
2014/04/29 职场文书
工厂搬迁方案
2014/05/11 职场文书
2014年公务员个人工作总结
2014/11/22 职场文书
军事理论课感想
2015/08/11 职场文书
2019年学校消防安全责任书(2篇)
2019/10/09 职场文书
Redis批量生成数据的实现
2022/06/05 Redis