我正在玩各种不同的方法来调用一个 Javascript中的Object的属性,并查看哪个类型的调用将’this’设置为Object,并将’this’设置为全局对象.

这是我的测试代码:

var foo = {
  bar: function(){ 
    console.log('this:' + this);
  }
}

console.log('calling foo.bar()');
foo.bar();

console.log('\ncalling (foo.bar)()');
(foo.bar)();

console.log('\ncalling f=foo; f.bar()');
f = foo; f.bar();

console.log('\ncalling f=foo.bar; f()');
f = foo.bar; f();

console.log('\ncalling (f=foo.bar)()');
(f = foo.bar)();

这就是结果:

calling foo.bar()
this:[object Object]

calling (foo.bar)()
this:[object Object]

calling f=foo; f.bar()
this:[object Object]

calling f=foo.bar; f()
this:[object global]

calling (f=foo.bar)()
this:[object global]

我的问题是,为什么f = foo.bar; F();和(f = foo.bar)();将’this’赋值为全局对象

解决方法

了解这个对函数调用的隐含行为的关键在于知道 Reference Type如何操作.

参考类型由两个组件(ECMAScript 3)组成,基础对象和属性名称(在ECMAScript 5中,它具有第三个组件严格标志 – 稍后将讨论严格模式(1)).

当一个function is invoked时,通过获取引用的基础对象(通过内部GetBase操作)来隐式地确定该值.

例如,在foo.bar引用中,基础对象为foo,属性名称为“bar”:

foo.bar(); // `this` will point to `foo`

执行作业时,引用将丢失,我们不再有基础对象和属性名称,我们只有一个值:

(f=foo.bar)(); // `this` will point to the global object

它不会发生只有分配,它发生与使用内部GetValue操作的其他操作:

// `this` will point to the global object    
(0,foo.bar)();   // The Comma Operator
(0 || foo.bar)(); // Binary Logical Operators
(1 && foo.bar)();
// etc..

例如,如果您用括号(正式称为The Grouping Operator)包围引用,则不会发生:

(foo.bar)(); // `this` will point to `foo`

分组运算符(再次,圆括号;)不在内部使用GetValue,这是以这种方式设计的,因为typeof和delete运算符被允许使用括号表达式:

delete (foo.bar);

如果分组运算符使用GetValue,那么delete运算符将无法获取一个基础对象去除属性栏.

回到隐含的这个值,有一些棘手的例子,例如使用with语句:

with (foo) {
  bar(); // `this` will refer to `foo` !
}

你可以看到,调用bar();在with块中,仍然将该值绑定到foo对象.

这个值不是从引用中设置的,它来自当前的环境记录(我可能稍后会写一些关于它的内容),由with语句引入.

另外,您可能会注意到,对于非引用,该值将指向全局对象,例如:

(function () {})(); // this will point to the global object

我们只是一个值,而不是一个引用(记住,一个引用是一个已解析的名称绑定).

(1)注意:在ECMAScript 5严格模式下,对于所有这些值隐含设置为全局对象的情况,该值将不被定义.

这是作为一种安全措施,主要是由于人们经常忘记在调用构造函数时使用新的运算符,从而对全局范围造成不良的行为和污染.

例如:

function Foo () {
  this.prop = 'foo';
}
Foo(); // no `new` operator,boom!

如你所知,Foo引用没有直接的基础对象,这将指向全局对象,它将无意中创建一个属性.

在严格的模式下,代码只会给你一个TypeError,因为这个值将是未定义的.

同样你可能记得在一开始我提到参考类型有一个第三个组件,严格的标志.

你原来的例子:

(f=foo.bar)();

可能甚至不会在严格模式下工作,因为未声明的标识符(如f似乎)的分配是不允许的(另一种安全措施来避免全球对象污染).

更多信息:

> Strict Mode

例如“this”是Javascript中的全局对象的更多相关文章

  1. 关于h5中的fetch方法解读(小结)

    这篇文章主要介绍了关于h5中的fetch方法解读(小结),fetch身为H5中的一个新对象,他的诞生,是为了取代ajax的存在而出现,有兴趣的可以了解一下

  2. 如何在iOS 10上设置日志级别?

    换句话说,如果我在iOS上运行的代码就像这样:那么在Console.app中看到记录的消息需要做些什么呢?

  3. 为什么Xcode 8(iOS 10)在控制台中打印[LogMessageLogging]

    为什么Xcode8打印[LogMessageLogging]在控制台中,当我调用地图视图时?任何人都可以提出一些建议吗?解决方法PrivacyTheunifiedloggingsystemconsidersdynamicstringsandcomplexdynamicobjectstobeprivate,anddoesnotcollectthemautomatically.Toensuretheprivacyofusers,itisrecommendedthatlogmessagesconsiststri

  4. ios – 在Swift 4中为os_log传递可变参数

    我正在尝试为Swift4/iOS11中的os_log编写一个方便的包装器,但是我已经遇到了传递可变参数的艰难战斗.基本上,我想编写一个如下所示的函数.不幸的是,我似乎无法弄清楚传递参数的神奇语法,并且在CVararg讨论的泥潭中有点迷失.(…这让我想念Python的splatting语法)解决方法我还没有找到解决方案,所以这个愚蠢的黑客:

  5. xcode – osx上的config.log是什么?它在哪里?

    任何人都可以解释’configure’是什么和做什么,一般可以找到config.log文件?

  6. api – HTTPS请求仅在iOS,Ionic 2上失败

    我有一个Ionic2应用程序,它调用SpringBootAPI将推送通知发送到其他设备.API使用HTTPS配置.APIPOST请求适用于除iOS之外的所有内容.我在服务器上的SSL证书是自签名的(可能就是这样吗?

  7. swift学习日志—— Log日志

    Log输出是程序开发中很重要的组成部分,虽然它并不是直接的业务代码,但是却可以忠实地反映我们的程序是如何工作的,以及记录程序运行的过程中发生了什么。在OC中的Log日志设置请看我的另一篇博客:设置Log日志打印开关在Swift中,最简单的输出方法就是使用print,在我们关心的地方输出字符串和值。如果我们在开发中就注意使用了统一的log输出的话,这就变得非常简单了。

  8. 使用XcodeColors 来显示XCGLogger,进行swift 的logger定制

    XcodeColors项目地址XcodeColorsinstallationinstructionsforXcode4,5,6&7:Downloadorclonetherepository.OpentheXcodeColorsprojectwithXcodeIfcompilingforXcode4,thenchangetheschemestousetheXcode4buildconfigurati

  9. swift 自定义log输出

    swift自定义log输出直接上代码

  10. swift LOG 输出

    在Swift中,最简单的输出方法就是使用print,在我们关心的地方输出字符串和值。最棒的是,我们不再需要对这样的输出进行维护,无论在哪里它都能正确地输出各个参数://...printLog//...}//输出://Test.swift[62],method():这是一条输出另外,对于log输出更多地其实是用在程序开发和调试的过程中的,过多的输出有可能对运行的性能造成影响。在Release版本中关闭掉向控制台的输出也是软件开发中一种常见的做法。

随机推荐

  1. js中‘!.’是什么意思

  2. Vue如何指定不编译的文件夹和favicon.ico

    这篇文章主要介绍了Vue如何指定不编译的文件夹和favicon.ico,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教

  3. 基于JavaScript编写一个图片转PDF转换器

    本文为大家介绍了一个简单的 JavaScript 项目,可以将图片转换为 PDF 文件。你可以从本地选择任何一张图片,只需点击一下即可将其转换为 PDF 文件,感兴趣的可以动手尝试一下

  4. jquery点赞功能实现代码 点个赞吧!

    点赞功能很多地方都会出现,如何实现爱心点赞功能,这篇文章主要为大家详细介绍了jquery点赞功能实现代码,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

  5. AngularJs上传前预览图片的实例代码

    使用AngularJs进行开发,在项目中,经常会遇到上传图片后,需在一旁预览图片内容,怎么实现这样的功能呢?今天小编给大家分享AugularJs上传前预览图片的实现代码,需要的朋友参考下吧

  6. JavaScript面向对象编程入门教程

    这篇文章主要介绍了JavaScript面向对象编程的相关概念,例如类、对象、属性、方法等面向对象的术语,并以实例讲解各种术语的使用,非常好的一篇面向对象入门教程,其它语言也可以参考哦

  7. jQuery中的通配符选择器使用总结

    通配符在控制input标签时相当好用,这里简单进行了jQuery中的通配符选择器使用总结,需要的朋友可以参考下

  8. javascript 动态调整图片尺寸实现代码

    在自己的网站上更新文章时一个比较常见的问题是:文章插图太宽,使整个网页都变形了。如果对每个插图都先进行缩放再插入的话,太麻烦了。

  9. jquery ajaxfileupload异步上传插件

    这篇文章主要为大家详细介绍了jquery ajaxfileupload异步上传插件,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

  10. React学习之受控组件与数据共享实例分析

    这篇文章主要介绍了React学习之受控组件与数据共享,结合实例形式分析了React受控组件与组件间数据共享相关原理与使用技巧,需要的朋友可以参考下

返回
顶部