1、匿名函数

函数是JavaScript中最灵活的一种对象,这里只是讲解其匿名函数的用途。匿名函数:就是没有函数名的函数。

1.1 函数的定义,首先简单介绍一下函数的定义,大致可分为三种方式

第一种:这也是最常规的一种

  1. function double(x){
  2. return 2 * x;
  3. }

第二种:这种方法使用了Function构造函数,把参数列表和函数体都作为字符串,很不方便,不建议使用。

  1. var double = new Function('x', 'return 2 * x;');

第三种:

  1. var double = function(x) { return 2* x; }

注意“=”右边的函数就是一个匿名函数,创造完毕函数后,又将该函数赋给了变量square。

1.2 匿名函数的创建

第一种方式:就是上面所讲的定义square函数,这也是最常用的方式之一。

第二种方式:

  1. (function(x, y){
  2. alert(x + y);
  3. })(2, 3);

这里创建了一个匿名函数(在第一个括号内),第二个括号用于调用该匿名函数,并传入参数。

2、闭包

闭包的英文单词是closure,这是JavaScript中非常重要的一部分知识,因为使用闭包可以大大减少我们的代码量,使我们的代码看上去更加清晰等等,总之功能十分强大。

闭包的含义:闭包说白了就是函数的嵌套,内层的函数可以使用外层函数的所有变量,即使外层函数已经执行完毕(这点涉及JavaScript作用域链)。

示例一

  1. function checkClosure(){
  2. var str = 'rain-man';
  3. setTimeout(
  4. function(){ alert(str); } //这是一个匿名函数
  5. , 2000);
  6. }
  7. checkClosure();

这个例子看上去十分的简单,仔细分析下它的执行过程还是有许多知识点的:checkClosure函数的执行是瞬间的(也许用时只是0.00001毫秒),在checkClosure的函数体内创建了一个变量str,在checkClosure执行完毕之后str并没有被释放,这是因为setTimeout内的匿名函数存在这对str的引用。待到2秒后函数体内的匿名函数被执行完毕,str才被释放。

示例二,优化代码

  1. function forTimeout(x, y){
  2. alert(x + y);
  3. }
  4. function delay(x , y , time){
  5. setTimeout('forTimeout(' + x + ',' + y + ')' , time);
  6. }
  7. /**
  8. * 上面的delay函数十分难以阅读,也不容易编写,但如果使用闭包就可以让代码更加清晰
  9. * function delay(x , y , time){
  10. * setTimeout(
  11. * function(){
  12. * forTimeout(x , y)
  13. * }
  14. * , time);
  15. * }
  16. */

3、举例

匿名函数最大的用途是创建闭包(这是JavaScript语言的特性之一),并且还可以构建命名空间,以减少全局变量的使用。

示例三:

  1. var oEvent = {};
  2. (function(){
  3. var addEvent = function(){ /*代码的实现省略了*/ };
  4. function removeEvent(){}
  5.  
  6. oEvent.addEvent = addEvent;
  7. oEvent.removeEvent = removeEvent;
  8. })();

在这段代码中函数addEvent和removeEvent都是局部变量,但我们可以通过全局变量oEvent使用它,这就大大减少了全局变量的使用,增强了网页的安全性。 我们要想使用此段代码:oEvent.addEvent(document.getElementById('box') , 'click' , function(){});

示例四:

  1. var rainman = (function(x , y){
  2. return x + y;
  3. })(2 , 3);
  4. /**
  5. * 也可以写成下面的形式,因为第一个括号只是帮助我们阅读,但是不推荐使用下面这种书写格式。
  6. * var rainman = function(x , y){
  7. * return x + y;
  8. * }(2 , 3);
  9. */

在这里我们创建了一个变量rainman,并通过直接调用匿名函数初始化为5,这种小技巧有时十分实用。

示例五:

  1. var outer = null;
  2.  
  3. (function(){
  4. var one = 1;
  5. function inner (){
  6. one += 1;
  7. alert(one);
  8. }
  9. outer = inner;
  10. })();
  11.  
  12. outer(); //
  13. outer(); //
  14. outer(); //

这段代码中的变量one是一个局部变量(因为它被定义在一个函数之内),因此外部是不可以访问的。但是这里我们创建了inner函数,inner函数是可以访问变量one的;又将全局变量outer引用了inner,所以三次调用outer会弹出递增的结果。

4、注意

4.1 闭包允许内层函数引用父函数中的变量,但是该变量是最终值

示例六:

  1. /**
  2. * <body>
  3. * <ul>
  4. * <li>one</li>
  5. * <li>two</li>
  6. * <li>three</li>
  7. * <li>one</li>
  8. * </ul>
  9. */
  10.  
  11. var lists = document.getElementsByTagName('li');
  12. for(var i = 0 , len = lists.length ; i < len ; i++){
  13. lists[ i ].onmouseover = function(){
  14. alert(i);
  15. };
  16. }

你会发现当鼠标移过每一个<li&rt;元素时,总是弹出4,而不是我们期待的元素下标。这是为什么呢?注意事项里已经讲了(最终值)。显然这种解释过于简单,当mouseover事件调用监听函数时,首先在匿名函数( function(){ alert(i); })内部查找是否定义了 i,结果是没有定义;因此它会向上查找,查找结果是已经定义了,并且i的值是4(循环后的i值);所以,最终每次弹出的都是4。

解决方法一:

  1. var lists = document.getElementsByTagName('li');
  2. for(var i = 0 , len = lists.length ; i < len ; i++){
  3. (function(index){
  4. lists[ index ].onmouseover = function(){
  5. alert(index);
  6. };
  7. })(i);
  8. }

解决方法二:

  1. var lists = document.getElementsByTagName('li');
  2. for(var i = 0, len = lists.length; i < len; i++){
  3. lists[ i ].$$index = i; //通过在Dom元素上绑定$$index属性记录下标
  4. lists[ i ].onmouseover = function(){
  5. alert(this.$$index);
  6. };
  7. }

解决方法三:

  1. function eventListener(list, index){
  2. list.onmouseover = function(){
  3. alert(index);
  4. };
  5. }
  6. var lists = document.getElementsByTagName('li');
  7. for(var i = 0 , len = lists.length ; i < len ; i++){
  8. eventListener(lists[ i ] , i);
  9. }
  1. function forTimeout(x, y){
  2. alert(x + y);
  3. }
  4. function delay(x , y , time){
  5. setTimeout('forTimeout(' + x + ',' + y + ')' , time);
  6. }
  1.  
  1. 上面的delay函数十分难以阅读,也不容易编写,但如果使用闭包就可以让代码更加清晰
  1. function delay(x , y , time){
  2. setTimeout(
  3. function(){
  4. forTimeout(x , y)
  5. }
  6. , time);
  7. }

javascript匿名函数及闭包深入理解及应用的更多相关文章

  1. 第一百一十节,JavaScript匿名函数和闭包

    JavaScript匿名函数和闭包 学习要点: 1.匿名函数 2.闭包 匿名函数就是没有名字的函数,闭包是可访问一个函数作用域里变量的函数.声明:本节内容需要有面向对象和少量设计模式基础,否则无法听懂 ...

  2. php匿名函数和闭包的理解

    PHP匿名函数和闭包使用的句法与普通函数相同,但匿名函和闭包数其实是伪装成函数的对象. 匿名函数:就是没有名称的函数.匿名函数可以赋值给变量,对象传递.不过匿名函数仍是函数,因此可以调用,还可以传入参 ...

  3. 深入理解javascript 匿名函数和闭包

    代码如下: (function(){ //这里忽略jQuery所有实现 })(); (function(){ //这里忽略jQuery所有实现 })();  半年前初次接触jQuery的时候,我也像其 ...

  4. [从jQuery看JavaScript]-匿名函数与闭包(Anonymous Function and Closure)【转】

    (function(){ //这里忽略jQuery所有实现 })(); 半年前初次接触jQuery的时候,我也像其他人一样很兴奋地想看看源码是什么样的.然而,在看到源码的第一眼,我就迷糊了.为什么只有 ...

  5. Javascript 匿名函数与闭包

    请见如下一个闭包示例: color = "red"; var obj = { color: "blue", getColor: function () { fu ...

  6. [从jQuery看JavaScript]-匿名函数与闭包

    jQuery片段: (function(){ //这里忽略jQuery所有实现 })(); 半年前初次接触jQuery的时候,我也像其他人一样很兴奋地想看看源码是什么样的.然而,在看到源码的第一眼,我 ...

  7. javascript 匿名函数及闭包----转载

    网上很多解释,我无法理解,我想知道原理...这篇文章应该可以透彻一点Query片段:view plaincopy to clipboardprint? (function(){      //这里忽略 ...

  8. [从jQuery看JavaScript]-匿名函数与闭包(Anonymous Function and Closure)

    http://blog.csdn.net/natineprince/article/details/4759533   jQuery片段: (function(){ //这里忽略jQuery所有实现 ...

  9. 浅谈JavaScript匿名函数与闭包

    一. 匿名函数   //普通函数定义: //单独的匿名函数是无法运行的.就算运行了,也无法调用,因为没有名称. 如: function(){             alert('123');    ...

随机推荐

  1. 【DEBUG】不能将参数 1 从“CString”转换为“const char *”

    1.  在vc6.0下用CString str;num = atoi(str);就可以顺利取到num: 但是同样代码拿到vs2008就报错,error C2664: "atoi": ...

  2. [转]十五天精通WCF——第八天 对“绑定”的最后一点理解

    转眼已经中断10几天没有写博客了,也不是工作太忙,正好碰到了端午节,然后最近看天津台的爱情保卫战入迷了...太好看了,一直都是回味无穷...而且 涂磊老师话说的真是tmd的经典,然后就这样耽搁了,好了 ...

  3. iOS:如何让xib同时兼容支持iOS6和iOS7

    做法如下: 1. 取消xib的Use Autolayout属性的勾选: 2. 将xib中每一个子控件(没错,每一个控件,包括UIButton.UILabel等等),将它们的Y Frame的值全部增加2 ...

  4. Tween公式

    Tween公式 4个参数 t:current time(当前时间) b:beginning value(初始值) c: change in value(变化量) d:duration(持续时间) re ...

  5. UVA 11825 状态压缩DP+子集思想

    很明显的状态压缩思想了.把全集分组,枚举每个集合的子集,看一个子集是否能覆盖所有的点,若能,则f[s]=max(f[s],f[s^s0]+1).即与差集+1比较. 这种枚举集合的思想还是第一次遇到,果 ...

  6. clear out all variables without closing terminal

    clear out all variables without closing terminal https://unix.stackexchange.com/questions/172655/cle ...

  7. $().attr()的使用方法 &amp;&amp; $().html()与$().text()的差别

    <1>$().attr()的使用方法 </pre><pre class="html" name="code"><htm ...

  8. 多工程联编的Pods如何设置

    多工程联编的Pods如何设置 (2014-07-17 13:57:10) 转载▼ 标签: 联编 多工程 分类: iOS开发 如今,CocoaPods使用越来越多,几乎每个项目都会使用到.有时候我们的项 ...

  9. JQuery实现复制到剪贴板功能

    在网页中实现复制到剪贴板功能,有两种方法, 第1种方法:使用JavaScript自带的方法,但是这种方法只能在IE下使用. document.execCommand("Copy") ...

  10. Django网站管理--ModelAdmin

    class AuthorAdmin(admin.ModelAdmin): list_display=('name', 'age', 'sex') #指定要显示的字段 search_fields=('n ...