1.判断原型对象 isPrototypeOf
使用 obj.__proto__==Array.prototype
问题: __proto__是隐藏属性,个别浏览器禁用
解决: 父对象.isPrototypeOf(子对象)
var obj1 = { name: "carmen" }, obj2 = [1,3,5,7];
console.log(
obj1.__proto__ == Array.prototype,//false
obj2.__proto__ == Array.prototype,//true
Array.prototype.isPrototypeOf(obj1),//false
Array.prototype.isPrototypeOf(obj2)//true
);
2.判断构造函数 instanceof
使用 obj.constructor==Array
问题: constructor可能禁止使用
解决: obj instanceof Array
问题: 不够严格
(1). __proto__和constructor不止检查直接父级,且检查整个原型链。
(2). __proto__有可能被修改!
var obj1 = { name: "carmen" }, obj2 = [1,3,5,7];
console.log(
//constructor可能禁止使用
obj1.constructor == Array,//false
obj2.constructor == Array,//true
// 使用instanceof
obj1 instanceof Array,//false
obj2 instanceof Array,//true
);
// __proto__被修改情况,不严谨
var obj3 = new Date();
obj3.__proto__ = Array.prototype;
console.log(obj3 instanceof Array);//true
3.判断隐藏的class属性
class属性: 隐藏在每个对象内部,记录对象创建时的类型名。不随继承关系改变而改变。
问题: class属性是内部隐藏属性,不能用.直接访问!
解决: 只有顶级父类型Object.prototype中的toString()才可输出class属性值
问题: 内置类型的prototype中都重写了新的toString()覆盖了顶级的toString()
解决: 特殊的函数: .call()
var obj1 = { name: "carmen" }, obj2 = [1,3,5,7];
console.log(
obj2.toString(), // 1,3,5,7 不是预期结果
Object.prototype.toString.call(obj1),//[object Object]
Object.prototype.toString.call(obj2), // [object Array]
Object.prototype.toString.call(obj1) == "[object Array]",// false
Object.prototype.toString.call(obj2) == "[object Array]" //true
);
- ES5新API:
var bool=Array.isArray(obj);
强调: 其实采用的也是第三种方法
console.log(
Array.isArray(obj1),// flase
Array.isArray(obj2),// true
Array.isArray(obj3) // false obj3在第2种方法中结果为true,其实不是Array类型
);