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

  请看题:

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. WCF入门(22)

    前言 本还想写一集WCF入门教程的,心情实在不好,明天又还有面试,改天再写吧. 说一下今天遇到的入职坑.面试能坑,上班能坑,完全没想到入职也能坑.切身经历. 今年10月份想换工作,更新了一下简历,接到 ...

  2. Bata版本冲刺计划及安排

    下一阶段需要改进完善的功能 UI: 1. 界面不够精细,布局不够美观并且尚有BUG没有解决,图形.色彩间不够协调. 2. 理清界面间的跳转逻辑. 搜索: 搜索成功并定位后,不会影响地理标签的显示. 定 ...

  3. java模板和回调机制学习总结

    最近看spring的JDBCTemplete的模板方式调用时,对模板和回调产生了浓厚兴趣,查询了一些资料,做一些总结. 回调函数: 所谓回调,就是客户程序C调用服务程序S中的某个函数A,然后S又在某个 ...

  4. Ibatis学习总结3--SQL Map XML 映射文件

    在前面的例子中,只使用了 SQL Map 最简单的形式.SQL Map 的结构中还有其他更多 的选项.这里是一个 mapped statement 较复杂的例子,使用了更多的特性. <sqlMa ...

  5. 一个项目中哪些文件是要上传到 git上的,哪些是不必要的

  6. POJ3749 破译密码

    Description 据说最早的密码来自于罗马的凯撒大帝.消息加密的办法是:对消息原文中的每个字母,分别用该字母之后的第5个字母替换(例如:消息原文中的每个字母A都分别替换成字母F).而你要获得消息 ...

  7. IPC机制

    转:http://blog.chinaunix.net/uid-26125381-id-3206237.html  IPC 三种通信机制 2012-05-13 17:23:55 最近看了,IPC三种通 ...

  8. 导师互选系统 Alpha版冲刺总结

    导师互选系统 Alpha版冲刺总结 一.设想和目标 我们的软件什么问题?是否定义得很清楚?是否对典型用户和典型场景有清晰的描述? 我们的软件主要是要实现导师和学生双向互选的功能.功能定义清晰明确,在软 ...

  9. MyEclipse------制作通讯录

    addinfo.java public class addinfo extends HttpServlet { private String url="jdbc:mysql://localh ...

  10. User表格式

    "_id":基本是700多 "name":"xx01" "pwd":"123"