谈谈 Object.prototype.toString 。
栏目: JavaScript · 发布时间: 7年前
内容简介:谈谈 Object.prototype.toString 。
原文链接 我的blog 。
前几日看到一个比较熟悉的面试题,判断一个变量是不是数组?
以下几种方法供参考:
var arr = [1, 2, 3] Array.isArray(arr) arr instanceof Array arr.constructor === Array Object.prototype.toString.call(arr) === '[object Array]' ...
这篇文章主要是谈谈 Object.prototype.toString
。
ECMAScript 5
在ECMAScript 5中, Object.prototype.toString()
被调用时,会进行如下步骤:
-
如果
this
是undefined
,返回[object Undefined]
; -
如果
this
是null
, 返回[object Null]
; -
令
O
为以this
作为参数调用ToObject
的结果; -
令
class
为O
的内部属性[[Class]]
的值; -
返回三个字符串
"[object", class, 以及"]"
拼接而成的字符串。
[[Class]]
[[Class]]
是一个内部属性,值为一个类型字符串,可以用来判断值的类型。
有这么一段详细的解释:
本规范的每种内置对象都定义了 [[Class]] 内部属性的值。宿主对象的 [[Class]] 内部属性的值可以是除了 "Arguments", "Array", "Boolean", "Date", "Error", "Function", "JSON", "Math", "Number", "Object", "RegExp", "String" 的任何字符串。[[Class]] 内部属性的值用于内部区分对象的种类。注,本规范中除了通过 Object.prototype.toString ( 见 15.2.4.2) 没有提供任何手段使程序访问此值。
在JavaScript代码里,唯一可以访问该属性的方法就是通过 Object.prototype.toString
,通常方法如下:
Object.prototype.toString.call(value)
举例:
> Object.prototype.toString.call(null) '[object Null]' > Object.prototype.toString.call(undefined) '[object Undefined]' > Object.prototype.toString.call(Math) '[object Math]' > Object.prototype.toString.call({}) '[object Object]' > Object.prototype.toString.call([]) '[object Array]'
因此,可以用下列函数,来获取任意变量的 [[Class]]
属性:
function getClass (a) { const str = Object.prototype.toString.call(a) return /^\[object (.*)\]$/.exec(str)[1] }
运行即可得
> getClass(null) 'Null' > getClass(undefined) 'Undefined' > getClass(Math) 'Math' > getClass({}) 'Object' > getClass([]) 'Array'
ECMAScript 6
在ES6,调用 Object.prototype.toString
时,会进行如下步骤:
-
如果
this
是undefined
,返回'[object Undefined]'
; -
如果
this
是null
, 返回'[object Null]'
; -
令
O
为以this
作为参数调用ToObject
的结果; -
令
isArray
为IsArray(O)
; -
ReturnIfAbrupt(isArray)
(如果isArray
不是一个正常值,比如抛出一个错误,中断执行); -
如果
isArray
为true
, 令builtinTag
为'Array'
; -
else
,如果O is an exotic String object
, 令builtinTag
为'String'
; -
else
,如果O
含有[[ParameterMap]] internal slot,
, 令builtinTag
为'Arguments'
; -
else
,如果O
含有[[Call]] internal method
, 令builtinTag
为Function
; -
else
,如果O
含有[[ErrorData]] internal slot
, 令builtinTag
为Error
; -
else
,如果O
含有[[BooleanData]] internal slot
, 令builtinTag
为Boolean
; -
else
,如果O
含有[[NumberData]] internal slot
, 令builtinTag
为Number
; -
else
,如果O
含有[[DateValue]] internal slot
, 令builtinTag
为Date
; -
else
,如果O
含有[[RegExpMatcher]] internal slot
, 令builtinTag
为RegExp
; -
else
, 令builtinTag
为Object
; -
令
tag
为Get(O, @@toStringTag)
的返回值(Get(O, @@toStringTag)
方法,既是在O
是一个对象,并且具有@@toStringTag
属性时,返回O[Symbol.toStringTag]
); -
ReturnIfAbrupt(tag)
,如果tag
是正常值,继续执行下一步; -
如果
Type(tag)
不是一个字符串,let tag be builtinTag
; -
返回由三个字符串
"[object", tag, and "]"
拼接而成的一个字符串。
在ES6里,之前的 [[Class]]
不再使用,取而代之的是一系列的 internal slot
,有一个比较完整的解释:
Internal slots correspond to internal state that is associated with objects and used by various ECMAScript specification algorithms. Internal slots are not object properties and they are not inherited. Depending upon the specific internal slot specification, such state may consist of values of any ECMAScript language type or of specific ECMAScript specification type values
大概的意思是:Internal slots 对应于与对象相关联并由各种ECMAScript规范算法使用的内部状态,它们没有对象属性,也不能被继承,根据具体的 Internal slot 规范,这种状态可以由任何ECMAScript语言类型或特定ECMAScript规范类型值的值组成。
此外,通过对 Object.prototype.toString
在ES6的实现步骤分析,我们其实可以很容易改变 Object.prototype.toString.call
的结果,像下面一样:
let obj = {} Object.defineProperty(obj, Symbol.toStringTag, { get: function() { return "newClass" } }) console.log(Object.prototype.toString.call(obj)) // "[object newClass]"
ECMAScript 7
ES7目前还是工作草案,到目前为止,就 Object.prototype.toString
的实现步骤来说, 只是移除了ES6其中的 ReturnIfAbrupt
。
参考
- http://www.ecma-international.org/ecma-262/5.1
- http://www.adobe.com/devnet/archive/html5/articles/categorizing-values-in-javascript.html
- https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Object/toString
- http://www.ecma-international.org/ecma-262/6.0/
- http://es6.ruanyifeng.com/#docs/symbol
- https://tc39.github.io/ecma262/#sec-object.prototype.tostring
完。
以上就是本文的全部内容,希望本文的内容对大家的学习或者工作能带来一定的帮助,也希望大家多多支持 码农网
猜你喜欢:本站部分资源来源于网络,本站转载出于传递更多信息之目的,版权归原作者或者来源机构所有,如转载稿涉及版权问题,请联系我们。
计算机程序设计艺术:第4卷 第4册(双语版)
Donald E.Knuth / 苏运霖 / 机械工业出版社 / 2007-4 / 42.00元
关于算法分析的这多卷论著已经长期被公认为经典计算机科学的定义性描述。迄今已出版的完整的三卷组成了程序设计理论和实践的惟一的珍贵源泉,无数读者都赞扬Knuth的著作对个人的深远影响。科学家们为他的分析的美丽和优雅所惊叹,而从事实践的程序员们已经成功地应用他的“菜谱式”的解到日常问题上,所有人都由于Knuth在书中所表现出的博学、清晰、精确和高度幽默而对他无比敬仰。 为开始后续各卷的写作并更......一起来看看 《计算机程序设计艺术:第4卷 第4册(双语版)》 这本书的介绍吧!