nodejs中this关键字的问题

再分析具体内容之前,必须要好好阅读以下以下两篇blog

学习Javascript闭包(closure)

Javascript的this用法

这两篇文章是阮一峰老师对Javascript的闭包和this用法的总结。

总结来说,闭包可以大概的理解为执行函数对环境上下文的变量的绑定。this是指向调用函数的环境对象。

第二篇中,this的用法阮老师是相对浏览器进行的分析。在评论中很多同学已经指出nodejs中this的关键字并非文中那样的属性。

亲自实验之后发现确实如评论所说的。

(1)首先针对阮老师第二篇中第4段代码,在nodejs中片段如下:

var x = 1;
function test(){
  this.x = 0;
}
test();
console.log(x); //1而非0

在阮老师的浏览器脚本环境中,阮老师说返回为0,也就是说this指向的是当前的window对象。

在nodejs中,当调用test()后,本代码片段所在的执行模块的全局x被置为1,此处用console输出的x仍是var声明的x。可以通过修改一下上面的代码得到验证。

x = 1;
function test(){
  this.x = 0;
}
console.log(x); //返回1
test();
console.log(x); //返回0

(2)分析闭包篇中阮老师最后留给大家分析的代码

首先把代码贴出来:

var object = {
  name : "My Object",
  getNameFunc : function() {
    console.log(this.name);
    return function () {
      return this.name;
    };
  }
};
console.log(object.getNameFunc()());


得到的结分别是My object 和 undefined.  这时估计很多同学就不能理解,前面的输出是来自于闭包的使用,那后面的函数返回中this对象到底来自于哪里?this不是应该指代调用该函数的那个环境对象吗?那个环境对应应该是什么呢?

其实对代码稍作改动,就会得到一个初步的答案。

将第一行的代码 var name = “local"; 改为 name = “global";

我们就会发现第二个的结果有了输出,global 。相信部分同学心里已经有了答案,那我们验证真正的原因。再次改造一下代码:

var object = {
  name : "My Object",
  getNameFunc : function(){
    return function () {
      return this.name;
    };
  }
};
name = "global";
console.log(object.getNameFunc()());
var obj = {
  name : "last result",
  myf : object.getNameFunc()
};
console.log(obj.myf());

nodejs运行后,分别得到输出 global 和 last result。

那答案已经非常清楚。this关键字是在真正被执行到的时候才会发挥作用。this对象仍然是指代函数被执行时执行该函数的对象。

当我把object.getNameFunc()赋值给myf是,实际等价于将函数function (){ return this.name; }赋值给myf,当myf被调用的时候,this由执行确定指向了对象obj。

所以我们总结来说:

同学们都被阮老师忽悠了!!!但是Javascript中闭包和this都是很关键的知识,需要完全掌握,否则很容易出错。

时间: 2024-08-01 10:22:47

nodejs中this关键字的问题的相关文章

在NodeJS中玩转Protocol Buffer

Protocol Buffer入门教程 Protocol Buffer是个什么鬼 NodeJS开发者为何要跟Protocol Buffer打交道 在NodeJS中实践Protocol Buffer协议 选择支持protobuf的NodeJS第三方模块 一个栗子 书写proto文件 编译 proto 文件 编写 Writer 编写Reader 运行结果 再举一个栗子 编写proto 编写client 书写server 运行结果 其他高级特性 嵌套Message Import Message 总结一

笔记-Nodejs中的核心API之Events

最近正在学习Node,在图书馆借了基本关于Node的书,同时在网上查阅资料,颇有收获,但是整体感觉对Node的理解还是停留在一个很模棱两可的状态.比如Node中的模块,平时练习就接触到那么几个,其他的一些模块暂时只会在学习的时候接触到,不常用便就荒废了.正所谓好记心不如烂笔头,多做笔记还是更有利于理解和记忆.自己做的总结也方便回头复习,所以决定踏上漫长的修炼之旅-- Node提供了许多API,其中一些比较重要.这些核心的API是所有Node应用的支柱,你会不停的用到他们. Events 几乎所有

nodejs(第三篇):nodejs中的文件模块、nodejs中的require与exports、http模块补充、JavaScript Standard Style

一.nodejs中的文件模块 在nodejs中,用户自己编写的模块,称为文件模块. 文件模块,则是指js文件.json文件或者是.node文件.在引用文件模块的时候后要加上文件的路径:/.../.../xxx.js表示绝对路径../xxx.js表示相对路径(同一文件夹下的xxx.js),../表示上一级目录.如果既不加/.../.../又不加./的话,则该模块要么是核心模块,要么是从一个node_modules文件夹加载. (1)在Node.js中,require是一个方法,只有通过requir

PHP 面向对象中常见关键字使用(final、static、const和instanceof)

PHP 面向对象中常见关键字的使用: 00x1.Final :final关键字可以加在类或者类中方法之前,但是不能使用final标识成员属性. 作用: 使用final标识的类,不能被继承. 在类中使用final标识的成员方法,在子类中不能覆盖. 总结:final表示为最终的意思,所以使用final关键字的类或者类中的成员方法是不能被更改的. 00x2.Static :static关键字将类中的成员属性或者成员方法标识为静态的,static标识的成员属性属于整个类,static成员总是唯一存在的,

Node.js(十二)——NodeJs中的Promise

爬虫基于回调和事件的方式去实现,回调也是被诟病已久的问题尤其是callback这种,无论是阅读还是调试都很费劲,甚至我们连代码的堆栈都看不到,这是一种反人类的写法,Promise来拜托这种痛苦的方式 传统方式实现动画效果: <!DOCTYPE html> <html> <head> <meta charset="UTF-8"> <title>Promise animation</title> <style&g

C/C++中extern关键字详解

1 基本解释:extern可以置于变量或者函数前,以标示变量或者函数的定义在别的文件中,提示编译器遇到此变量和函数时在其他模块中寻找其定义.此外extern也可用来进行链接指定. 也就是说extern有两个作用,第一个,当它与"C"一起连用时,如: extern "C" void fun(int a, int b);则告诉编译器在编译fun这个函数名时按着C的规则去翻译相应的函数名而不是C++的,C++的规则在翻译这个函数名时会把fun这个名字变得面目 全非,可能是

c++中const关键字全面总结

一.const作用 1.const定义常量 注意:const只对它左边的东西起作用,唯一的例外就是const本身就是最左边的修饰符,那么它才会对右边的东西起作用. (1)const修饰变量,以下两种定义形式在本质上是一样的.它的含义是:const修饰的类型为TYPE的变量value是不可变的. TYPE const ValueName = value; const TYPE ValueName = value; (2)将const改为外部连接,作用于扩大至全局,编译时会分配内存,并且可以不进行初

转:java中volatile关键字的含义

转:java中volatile关键字的含义 在java线程并发处理中,有一个关键字volatile的使用目前存在很大的混淆,以为使用这个关键字,在进行多线程并发处理的时候就可以万事大吉. Java语言是支持多线程的,为了解决线程并发的问题,在语言内部引入了 同步块 和 volatile 关键字机制. synchronized 同步块大家都比较熟悉,通过 synchronized 关键字来实现,所有加上synchronized 和 块语句,在多线程访问的时候,同一时刻只能有一个线程能够用 sync

[转]nodejs中的process模块--child_process.exec

1.process是一个全局进程,你可以直接通过process变量直接访问它. process实现了EventEmitter接口,exit方法会在当进程退出的时候执行.因为进程退出之后将不再执行事件循环,所有只有那些没有回调函数的代码才会被执行. 在下面例子中,setTimeout里面的语句是没有办法执行到的. 1 process.on('exit', function () { 2 setTimeout(function () { 3 console.log('This will not ru