我有一个创建的a merging library递归合并对象。有时在中间会有一个实际上是特殊类的对象(比如Firestore的Timestamp)。
在我的合并函数中,我检查是否有某个对象是这样的:
function isObject (payload) {
const getType = Object.prototype.toString.call(payload).slice(8, -1)
return getType === 'Object'
}
但是,通过此检查,一些具有特殊原型的特殊类仍被视为常规JavaScript对象。
我的问题: 该对象将失去其特殊的类原型,因为我只通过非原型值进行递归合并。
我的问题: 如何更改上面的函数,不仅检查它是否是JavaScript对象,还检查它是否是常规JavaScript对象?
基本上我只想在true
上返回isObject(obj)
,如果obj
是这样的对象:{}
或任何道具。
但是一旦原型不同(因此它是一个特殊的类),那么我想在false
上返回isObject(obj)
除了检查原型,你可以检查它的constructor
是否是Object
:
const isPlainObject = obj => obj.constructor === Object && Object.getPrototypeOf(obj) === Object.prototype;
const obj1 = { foo: 'bar' };
const obj2 = new Date();
const obj3 = [];
[obj1, obj2, obj3].forEach(obj => console.log(isPlainObject(obj)));
请注意,使用getPrototypeOf
并检查它是===
到Object.prototype
比使用toString
更可靠 - 毕竟toString
可能会返回任何东西。
如果您可能将某些null或未定义的内容传递给isPlainObject
,那么当然要包括检查obj
是否真的在尝试访问其上的属性之前:
const isPlainObject = obj => obj && obj.constructor === Object && Object.getPrototypeOf(obj) === Object.prototype;
const obj1 = { foo: 'bar' };
const obj2 = new Date();
const obj3 = [];
[obj1, obj2, obj3].forEach(obj => console.log(isPlainObject(obj)));
它可能很棘手,因为一些javascript类型有一个像下面的typeof
“对象”
console.log(typeof []) // object
console.log(typeof null) // object
检查普通对象的一种简单方法是使用Object的toString
方法,并验证null
这样
/**@return boolean */
function isObject(val) {
const _toString = Object.prototype.toString;
return val !== null && _toString.call(val) === '[object Object]';
}
现在我们测试一下
isObject({}); // true
isObject({a: 2, b: 3}); // true
isObject(null); // false
isObject([]); // false
isObject(new Object()); // true
isObject(new Array()); // false