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. Axis2中使用wsdl2java.bat生成客户端代码

    1 准备环境 (1)下载Axis2的zip包axis2-1.5.5-bin.zip,并解压. 官方网址:http://ws.apache.org/axis2/ (2)设置环境变量(我的电脑->属 ...

  2. [转]十五天精通WCF——第一天 三种Binding让你KO80%的业务

    转眼wcf技术已经出现很多年了,也在.net界混的风生水起,同时.net也是一个高度封装的框架,作为在wcf食物链最顶端的我们所能做的任务已经简单的不能再简单了, 再简单的话马路上的大妈也能写wcf了 ...

  3. HDU 4514

    真是神奇,G++TLE,C++500MS... 判环有一个图论知识就是,m>=n时必有环.如果以m的范围建图,会MLE. 然后,利用拓扑排序再来判定是否有环,因为有些景点可能是孤立的.同时,在拓 ...

  4. 关东升的《从零開始学Swift》即将出版

    大家好: 苹果2015WWDC大会公布了Swift2.0,它较之前的版本号Swift1.x有非常大的变化.所以我即将出版<从零開始学Swift><从零開始学Swift>将在&l ...

  5. Android 组件ContentProvider

    Android 组件ContentProvider Android的数据存储有五种方式Shared Preferences.网络存储.文件存储.外储存储.SQLite,一般这些存储都仅仅是在单独的一个 ...

  6. React Native布局实践:开发京东client首页(三)——轮播图的实现

    上篇文章中,我们一起构建了京东client的TabBar.在本文中.将继续向大家介绍京东client首页轮播图及其下发功能button的开发方法,如今就让我们開始吧! 1.相关控件调研 眼下在Gith ...

  7. 一条SQL语句求前面记录的平均值

    有算法要求如下: For (i=1,i<=10,i++) { ta[i] = (t[1] + t[2] + ... + t[i]) / i; } 用一条SQL语句实现它: 分别用表变量 @ta ...

  8. P2932 [USACO09JAN]地震造成的破坏Earthquake Damage 爆搜

    这题怎么这么水~~~本来以为挺难的一道题,结果随便一写就过了...本来还不知道损坏的牛棚算不算,结果不明不白就过了... 题干: 农夫John的农场遭受了一场地震.有一些牛棚遭到了损坏,但幸运地,所有 ...

  9. Coursera Algorithms week2 基础排序 练习测验: Permutation

    题目原文: Given two integer arrays of size n , design a subquadratic algorithm to determine whether one ...

  10. 第2章 安装Nodejs 2-4 Linux下安装Nodejs

    linux下编译安装Nodejs  GCC和G++分别是GNU的C和C++编译器.它们在执行编译工作的时候把源代码通过预处理转化成汇编语言生成.i后缀的文件,再由汇编变成目标机器代码,最后连接目标代码 ...