(转载)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 相关文章推荐
Jquery进度条插件 Progress Bar小问题解决
Jul 12 Javascript
JS定时器实例
Apr 17 Javascript
jquery ajax同步异步的执行最终解决方案
Apr 26 Javascript
js实现类似于add(1)(2)(3)调用方式的方法
Mar 04 Javascript
jquery实现简单实用的弹出层效果代码
Oct 15 Javascript
jQuery+Ajax实现限制查询间隔的方法
Jun 07 Javascript
jQuery中clone()函数实现表单中增加和减少输入项
May 13 jQuery
js最实用string(字符串)类型的使用及截取与拼接详解
Apr 26 Javascript
浅谈Vuex注入Vue生命周期的过程
May 20 Javascript
vue中组件通信详解(父子组件, 爷孙组件, 兄弟组件)
Jul 27 Javascript
Vue实现计算器计算效果
Aug 17 Javascript
JavaScript实现简单计时器
Jun 22 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
老机欣赏|中国60年代精品收音机
2021/03/02 无线电
从零开始学YII2框架(二)通过 Composer 安装扩展插件
2014/08/20 PHP
php禁止直接从浏览器输入地址访问.php文件的方法
2014/11/04 PHP
php生成固定长度纯数字编码的方法
2015/07/09 PHP
PHP字典树(Trie树)定义与实现方法示例
2017/10/09 PHP
silverlight线程与基于事件驱动javascript引擎(实现轨迹回放功能)
2011/08/09 Javascript
javascript延时加载之defer测试
2012/12/28 Javascript
浅谈javascript中onbeforeunload与onunload事件
2015/12/10 Javascript
Javascript中的几种继承方式对比分析
2016/03/22 Javascript
基于JavaScript实现数码时钟效果
2020/03/30 Javascript
jQuery实现广告条滚动效果
2017/08/22 jQuery
vue接入腾讯防水墙代码
2019/05/07 Javascript
ES6 Map结构的应用实例分析
2019/06/26 Javascript
Python中struct模块对字节流/二进制流的操作教程
2017/01/21 Python
详解python中的json和字典dict
2018/06/22 Python
python3字符串操作总结
2019/07/24 Python
python cv2在验证码识别中应用实例解析
2019/12/25 Python
Tensorflow矩阵运算实例(矩阵相乘,点乘,行/列累加)
2020/02/05 Python
浅谈tensorflow模型保存为pb的各种姿势
2020/05/25 Python
详解python datetime模块
2020/08/17 Python
无需JS和jQuery代码实现CSS3鼠标浮动放大图片
2016/11/21 HTML / CSS
3种方式实现瀑布流布局小结
2019/09/05 HTML / CSS
巴西手表购物网站:eclock
2019/03/19 全球购物
豪华床上用品 :Jennifer Adams
2019/09/15 全球购物
意大利值得信赖的在线超级药房:PillolaStore
2020/02/05 全球购物
英语翻译系毕业生求职信
2013/09/29 职场文书
社团活动策划书范文
2014/01/09 职场文书
仓管岗位职责范本
2014/02/08 职场文书
数控专业毕业生自荐信范文
2014/03/04 职场文书
小学生三分钟演讲稿
2014/08/18 职场文书
国际残疾人日广播稿范文
2014/10/09 职场文书
2015年党风廉政建设责任书
2015/01/29 职场文书
2015年房产经纪人工作总结
2015/05/15 职场文书
纪录片信仰观后感
2015/06/08 职场文书
Win7/8.1用户可以免费升级到Windows 11系统吗?
2021/11/21 数码科技
十大最帅动漫男主 碓冰拓海上榜,第一是《灌篮高手》男主角
2022/03/18 日漫