今天在群里看到一个问题,让我纠结了好一会。下面是我的分析,感觉里面还有很多问题,关于作用域还是不太理解,希望大家看到问题第一时间反馈给我,看到实在受不了的地方说几句都没关系,谢谢。

  请看题:

1.对象字面量中fn1函数是立即执行的函数表达式。

            $(function(){
var number = 2;
var obj = {
number: 4,
fn1: (function(){
this.number *= 2;
number = number*2;
var number = 3;
return function(){
this.number *= 2;
number *= 3;
alert(number);
}
})()
};
var f = obj.fn1;
alert(number); /* 2 */
f(); /* 9 */
obj.fn1(); /* 27 */ /* 因为是自执行的,number值是可以保存下来的 */
alert(window.number); /* NaN */
          alert(obj.number); /* 8 */
}) </script>

  输出2, 9,27, NaN, 8。

  分析:

  首先看fn1这个函数,是立即执行的函数表达式,所以var f = obj.fn1的时候,已经访问fn1函数下的内容了。在立即执行函数表达式fn1中,this.number为NaN,因为在此访问不到自己的属性和方法(这个我也说不太清楚为什么,求解……);number = number * 2为NaN,因为number是undefined,这是根据预解析的缘故,会先定义var number; number= number *2; number = 3。接下来又定义了var number = 3,在该作用域中和内部作用域中,number暂时都是为3的。在return function(){}中,因为外部立即执行的函数表达式,所以this.number是有效的(求解……),number = number *3是根据外部作用域number值决定的。

  alert(number);输出2,因为预解析的缘故。

  f();输出9,因为外部number为3,所以3*3 = 9。

  obj.fn1();输出27,因为是立即执行函数,所有具有保存变量值的作用,number值是9,再用9*3 = 27,。

  alert(window.number);输出NaN,因为是在加载后输出,要让window.number有效,只能在局部作用域内才可以。

  alert(obj.number);输出8,因为fn1中this.number失效,return function()中this.number有用,所以只执行return 函数中的this.number *= 2。

2.对象字面量中fn1函数不是立即执行的函数表达式

            $(function(){
var number = 2;
var obj = {
number: 4,
fn1: function(){
this.number *= 2;
number = number*2;
var number = 3;
return function(){
this.number *= 2;
number *= 3;
alert(number);
}
}
};
var f = obj.fn1();
alert(number); /* 2 */
f(); /* 9 */
obj.fn1()(); /* 9 */ /* 这个和上面作用域不一样了,因为不是表达式,不是立即执行的,在新的作用域又全都是新的 */
alert(window.number); /* NaN */
alert(obj.number); /* 16 */ })

  输出2, 9, 9, NaN,16。

  分析:

  首先看fn1这个函数,它是一个函数引用(需调用才会执行),所以var f = obj.fn1()(后面需接括号)的时候,已经访问fn1函数下的内容了。在立即执行函数表达式fn1中,this.number为有效的,在一般函数可以访问到自己的属性和方法;number = number * 2为NaN,因为number是undefined,这是根据预解析的缘故,会先定义var number; number= number *2; number = 3。接下来又定义了var number = 3。在return function(){}中,因为外部函数为一般函数,所以this.number是无效的(求解……),number = number *3是根据外部作用域number值决定的。

  alert(number);输出2,因为预解析的缘故。

  f();输出9,因为外部number为3,所以3*3 = 9。

  obj.fn1()();输出9,因为它和f()不在同一作用域中,它调用了fn1()初始化number = 3,后面再输出一个obj.fn1()()还是为9。

  alert(window.number);输出NaN,因为是在加载后输出,要让window.number有效,只能在局部作用域内才可以。

  alert(obj.number);输出16,因为fn1中this.number有用,return function()中this.number失效,所以只执行fn1函数中的this.number *= 2。

对于(function(){}())和function(){}实例的作用域分析(里面有很多问题……)的更多相关文章

  1. function foo(){}、(function(){})、(function(){}())等函数区别分析

    前面一段时间,看到(function(){}),(function(){}())这些函数就犯晕,不知道它到底是什么意思,为什么函数外要加小括号,函数后要加小括号,加和不加到底有什么区别……一直犯迷糊, ...

  2. 转载 +function ($) { "use strict";}(window.jQuery);全面分析

    转载 https://www.cnblogs.com/cndotabestdota/p/5664112.html +function ($) { "use strict";}(wi ...

  3. +function ($) { "use strict";}(window.jQuery);全面分析

    +function ($) { "use strict"; }(window.jQuery); 怎么理解? 匿名函数闭包 我们先来理一理函数表达式和函数声明的区别 函数表达式: 函 ...

  4. javascript 中function(){},new function(),new Function(),Function 摘录

    函数是JavaScript中很重要的一个语言元素,并且提供了一个function关键字和内置对象Function,下面是其可能的用法和它们之间的关系. function使用方式 var foo01 = ...

  5. Javacript中(function(){})() 与 (function(){}()) 区别 {转}

    这个问题可以从不同的角度来看,但从结果上来说 :他们是一样的.首先,如果从AST(抽象语法树)的角度来看,两者的AST是一模一样的,最终结果都是一次函数调用.因此,就解析器产生的结果论而言,两者是没有 ...

  6. function(){}、var fun=function(){}和function fun(){}的区别

    一.基本定义 1.函数声明:使用function声明函数,并指定函数名. function fun() { // ...... } 2.函数表达式:使用function声明函数,但未指定函数名,将匿名 ...

  7. 【javascript】javascript中function(){},function(){}(),new function(){},new Function()

    和java比起来,javascript真的是松散的无以复加,不过这也让我们在无聊之余,有精力去探讨一些复杂的应用,从而在开发之路上,获得一些新的想法. javascript中的类的构造 javascr ...

  8. javascript 中function(){}(),new function(),new Function(),Function

    和java比起来,javascript真的是松散的无以复加,不过这也让我们在无聊之余,有精力去探讨一些复杂的应用,从而在开发之路上,获得一些新的想法. javascript中的类的构造 javascr ...

  9. function,new function,Function,new Function 之间的区别

    测试一: var fud01 = function()  { var temp = 100; this.temp = 200; return temp + this.temp; } alert(typ ...

随机推荐

  1. Use Windows Azure AD to create SSO projects

    Keywords Windows Azure AD, SSO Summary Use Windows Azure AD to create SSO projects Detailed Scenario ...

  2. [vijos1264]神秘的咒语(LCIS)

    描述身为拜月教的高级间谍,你的任务总是逼迫你出生入死.比如这一次,拜月教主就派你跟踪赵灵儿一行,潜入试炼窟底.据说试炼窟底藏着五行法术的最高法术:风神,雷神,雪妖,火神,山神的咒语.为了习得这些法术, ...

  3. iOS开发之格式化日期时间(转)

    在开发iOS程序时,有时候需要将时间格式调整成自己希望的格式,这个时候我们可以用NSDateFormatter类来处理.例如: //实例化一个NSDateFormatter对象 NSDateForma ...

  4. 2016 版 Laravel 系列入门教程(二)【最适合中国人的 Laravel 教程】

    本教程示例代码见: https://github.com/johnlui/Learn-Laravel-5 在任何地方卡住,最快的办法就是去看示例代码. 本篇文章中,我将跟宝宝们一起学习 Laravel ...

  5. 第九章:Java----泛型学习(最后过一遍)

    泛型:让集合记住里面元素的类型,避免取出时需要强制类型转换(大到小).   ClassCastException! 编译阶段就能发现错误.  语法更严格! 更不容易犯错! 1. 构造器的名字还是类名, ...

  6. zabbix 的学习应用之路

    1.zabbix  server的安装      http://www.cnblogs.com/smail-bao/p/5643136.html 2.zabbix  agent的安装        h ...

  7. php 判断是否get传值的参数是否存在

    if(is_array($_GET)&&count($_GET)>0)//先判断是否通过get传值了    {        if(isset($_GET["id&qu ...

  8. Java基础-ArrayList和LinkedList的区别

    大致区别:  1.ArrayList是实现了基于动态数组的数据结构,LinkedList基于链表的数据结构. 2.对于随机访问get和set,ArrayList觉得优于LinkedList,因为Lin ...

  9. Android中的Uri.parse()

    1,调web浏览器 Uri myBlogUri = Uri.parse("http://www.baidu.com"); returnIt = new Intent(Intent. ...

  10. iOS之单例

    今天在看多线程同步时,突然想到了单例的同步问题.自从dispatch_once出现后,我们创建单例非常简单且安全: static dispatch_once_t pred; static Single ...