为什么用Object.prototype.toString.call(obj)检测对象类型?

37 Essential JavaScript Interview Questions*,中有一道javascript题:

使用 typeof bar === "object" 检测 变量“bar”是否为对象有什么缺点?如何避免?

这是一个十分常见的问题,用 typeof 是否能准确判断一个对象变量,答案是否定的。null 的结果也是 objectArray 的结果也是 object,有时候我们需要的是 "纯粹" 的 object对象。如何避免呢?
比较好的方式是:

console.log(Object.prototype.toString.call(obj) === "[object Object]");

使用以上方式可以很好的区分各种类型:
(无法区分自定义对象类型,自定义类型可以采用 instanceof 区分)

console.log(Object.prototype.toString.call("jerry"));//[object String]
console.log(Object.prototype.toString.call(12));//[object Number]
console.log(Object.prototype.toString.call(true));//[object Boolean]
console.log(Object.prototype.toString.call(undefined));//[object Undefined]
console.log(Object.prototype.toString.call(null));//[object Null]
console.log(Object.prototype.toString.call({name: "jerry"}));//[object Object]
console.log(Object.prototype.toString.call(function(){}));//[object Function]
console.log(Object.prototype.toString.call([]));//[object Array]
console.log(Object.prototype.toString.call(new Date));//[object Date]
console.log(Object.prototype.toString.call(/\d/));//[object RegExp]

function Person(){};
console.log(Object.prototype.toString.call(new Person));//[object Object]

为什么这样就能区分呢?于是我去看了一下 toString 方法的用法:toString方法返回反映这个对象的字符串。
那为什么不直接用 obj.toString() 呢?

console.log("jerry".toString());//jerry
console.log((1).toString());//1
console.log([1,2].toString());//1,2
console.log(new Date().toString());//Wed Dec 21 2016 20:35:48 GMT+0800 (中国标准时间)
console.log(function(){}.toString());//function (){}
console.log(null.toString());//error
console.log(undefined.toString());//error

同样是检测对象 obj 调用 toString 方法(关于 toString方法的用法的可以参考 toString 的详解),obj.toString() 的结果和Object.prototype.toString.call(obj) 的结果不一样,这是为什么?

这是因为toStringObject的原型方法,而 Arrayfunction 等类型作为 Object的实例,都重写了 toString 方法。不同的对象类型调用 toString 方法时,根据原型链的知识,调用的是对应的重写之后的 toString 方法(function 类型返回内容为函数体的字符串,Array 类型返回元素组成的字符串.....),而不会去调用 Object 上原型 toString 方法(返回对象的具体类型),所以采用 obj.toString() 不能得到其对象类型,只能将 obj 转换为字符串类型;因此,在想要得到对象的具体类型时,应该调用 Object 上原型 toString 方法。

我们可以验证一下,将数组的 toString 方法删除,看看会是什么结果:

var arr=[1,2,3];
console.log(Array.prototype.hasOwnProperty("toString"));//true
console.log(arr.toString());//1,2,3
delete Array.prototype.toString;//delete操作符可以删除实例属性
console.log(Array.prototype.hasOwnProperty("toString"));//false
console.log(arr.toString());//"[object Array]"

删除了 ArraytoString 方法后,同样再采用 arr.toString ( ) 方法调用时,不再有屏蔽 Object 原型方法的实例方法,因此沿着原型链,arr最后调用了 ObjecttoString 方法,返回了和 Object.prototype.toString.call(arr) 相同的结果。因此想要准确判断一个对象变量,使用 typeof 是不可行的,这个时候应该使用Object.prototype.toString.call(obj)来获取更准确的类型。

你可能感兴趣的:(为什么用Object.prototype.toString.call(obj)检测对象类型?)