Object.prototype.toString.call(obj)的功能

可能作为程序员以及即将迈入程序员门槛的小伙伴都知道Object.prototype.toString.call(obj)方法,功能简单但是确是类型检测界的“扛把子”。 首先来讲,对刚接触的人来说,我们为什么要使用这种奇怪的方法呢?typeof也可以检测类型,然而它有什么优势?我们来引入一道JavaScript的题目来参考: 使用 typeof bar === "object" 检测 变量“bar”是否为对象有什么缺点?如何避免? 缺点很明确,typeof可以准确的检测bar是否为对象吗?答案是否定的。Array null…检测出来也均为object,因此规避方法还是要用到本文所提方法。接下来我们进行测试

 1`console.log(Object.prototype.toString.call("jerry"));//[object String]
 2console.log(Object.prototype.toString.call(12));//[object Number]
 3console.log(Object.prototype.toString.call(true));//[object Boolean]
 4console.log(Object.prototype.toString.call(undefined));//[object Undefined]
 5console.log(Object.prototype.toString.call(null));//[object Null]
 6console.log(Object.prototype.toString.call({name: "jerry"}));//[object Object]
 7console.log(Object.prototype.toString.call(function(){}));//[object Function]
 8console.log(Object.prototype.toString.call([]));//[object Array]
 9console.log(Object.prototype.toString.call(new Date));//[object Date]
10console.log(Object.prototype.toString.call(/\d/));//[object RegExp]
11
12function Person(){};
13console.log(Object.prototype.toString.call(new Person));//[object Object]`

这即是此方法的用处,它可以精准判断所传入参数的数据类型。

它是如何做到的呢?接下来让我们一起来探讨一下其原理

Object.prototype.toString.call(obj)类型检测原理

此方法为什么能检测数据类型呢?起初我也很奇怪,只是记住了此方法,并未深究,今天让我们来一探究竟。

首先,这句话的意思是让我们用Object原型上的toString方法作用在传入的obj的上下文中(通过call将this指向obj),那么我们知道数组本身也有toString()方法,那我们为什么非要用Object上的呢? 首先,让我们再来看一下toString()方法:

 1`var num = 123
 2num.toString() // '123'
 3
 4var str = 'hello'
 5str.toString() // 'hello'
 6
 7var bool = false
 8bool.toString() // 'false'
 9
10var arr = [1, 2, 3]
11arr.toString()  // '1,2,3'
12
13var obj = {lang:'zh'}
14obj.toString()  // '[object Object]'
15
16var fn = function(){}
17fn.toString()  // 'function(){}'
18
19null.toString()  // Cannot read property 'toString' of null
20
21undefined.toString() // Cannot read property 'toString' of undefined`

toString:由名字可以看出此方法是将传入的数据类型转换成字符串输出(null和undefined除外) 我们再来看一下Object以及其原型上的toString方法

 1`Object.toString()//"function Object() { [native code] }"
 2Object.prototype.toString()//"[object Object]"`

Object输出的是其函数体"function Object() { [native code] }"而Object上输出的是其类型。我们可以看出Object对象和它的原型链上各自有一个toString()方法,第一个返回的是一个函数,第二个返回的是值类型。

接下来让我们看一个表格:

数据类型例子return
字符串“foo”.toString()“foo”
数字1.toString()Uncaught SyntaxError: Invalid or unexpected token
布尔值false.toString()“false”
undefinedundefined.toString()Uncaught TypeError: Cannot read property ‘toString’ of undefined
nullnull.toString()Uncaught TypeError: Cannot read property ‘toString’ of null
StringString.toString()“function String() { [native code] }”
NumberNumber.toString()“function Number() { [native code] }”
BooleanBoolean.toString()“function Boolean() { [native code] }”
ArrayArray.toString()“function Array() { [native code] }”
FunctionFunction.toString()“function Function() { [native code] }”
DateDate.toString()“function Date() { [native code] }”
RegExpRegExp.toString()“function RegExp() { [native code] }”
ErrorError.toString()“function Error() { [native code] }”
PromisePromise.toString()“function Promise() { [native code] }”
ObejctObject.toString()“function Object() { [native code] }”
MathMath.toString()“[object Math]”
这就是各种数据类型调用toString()方法的返回值,由此我们看出不同的数据类型都有其自身toString()方法。所以上述toString()方法来自于Number、String、Boolean…等等这些类。
我们又知道,在JavaScript中,所有类都继承于Object,因此toString()方法应该也被继承了,但由上述可见事实并不像我们想的那样,其实各数据类型使用toString()后的结果表现不一的原因在于:所有类在继承Object的时候,改写了toString()方法。 Object原型上的方法是可以输出数据类型的。因此我们想判断数据类型时,也只能使用原始方法。继而有了此方法:Object.prototype.toString.call(obj)

接下来让我们进行验证

 1`// 定义一个数组
 2var arr = [1, 2, 3]
 3
 4// 数组原型上是否具有 toString() 方法
 5console.log(Array.prototype.hasOwnProperty('toString')) //true
 6
 7// 数组直接使用自身的 toString() 方法
 8console.log(arr.toString()) // '1,2,3'
 9
10// delete操作符删除数组原型上的 toString()
11delete Array.prototype.toString
12
13// 删除后,数组原型上是否还具有 toString() 方法
14console.log(Array.prototype.hasOwnProperty('toString')) //false
15
16// 删除后的数组再次使用 toString() 时,会向上层访问这个方法,即 Object 的 toString()
17console.log(arr.toString()) // '[object Array]'`

当我们把Array自身的toString()方法删除之后,再次使用它时,由原型链它会向上查找这个方法,即Object的toString(),也便将Object上的toString()方法作用在数组上,得出其数据类型[object Array]

至此便透彻了Object.prototype.toString.call(obj)功能及原理。

bdd068dfc8da3c17ef351177fbe653dd.png

个人笔记记录 2021 ~ 2025