下面两个语句产生相同的输出吗?我们有理由选择其中一种方式而不是另一种吗?

 if (key in object)

 if (object.hasOwnProperty(key))

当前回答

小心——它们不会产生相同的结果。

如果在原型链的某个地方找到key, in也会返回true,而Object。hasOwnProperty(就像名字已经告诉我们的那样),只会在key在该对象上直接可用时返回true(它的“拥有”属性)。

其他回答

小心——它们不会产生相同的结果。

如果在原型链的某个地方找到key, in也会返回true,而Object。hasOwnProperty(就像名字已经告诉我们的那样),只会在key在该对象上直接可用时返回true(它的“拥有”属性)。

我会用另一个例子来解释。 假设我们有以下两个属性的对象:

function TestObj(){
    this.name = 'Dragon';
}
TestObj.prototype.gender = 'male';

让我们创建TestObj实例:

var o = new TestObj();

让我们检查一下对象实例:

console.log(o.hasOwnProperty('name')); // true
console.log('name' in o); // true

console.log(o.hasOwnProperty('gender')); // false
console.log('gender' in o); // true

结论:

如果属性可被对象直接访问或从原型访问,则In运算符总是返回true hasOwnProperty()只在实例上存在属性,而在其原型上不存在时才返回true

如果我们想要检查原型上是否存在某些属性,逻辑上,我们会说:

console.log(('name' in o) && !o.hasOwnProperty('name')); //false
console.log(('gender' in o) && !o.hasOwnProperty('gender')); //true - it's in prototype

最后:

所以,关于这两个条件…

if (key in object)
if (object.hasOwnProperty(key))

...产生同样的结果,答案是显而易见的,视情况而定。

另一种形式(在中调用)枚举属性名(或键) 物体的。控件中的另一个属性名字符串 对象被赋值给变量。通常有必要进行测试 object.hasOwnProperty(变量)来确定属性名是否 是对象的真正成员,或者在原型链上找到。

 for (myvar in obj) {
     if (obj.hasOwnProperty(myvar)) { ... } }

(摘自Crockford的Javascript: The Good Parts)

in也会检查继承的属性,而hasOwnProperty则不是这样。

第一个版本更短(特别是在变量重命名的最小化代码中)

a in b

vs

b.hasOwnProperty(a)

不管怎样,正如@AndreMeinhold所说,它们并不总是产生相同的结果。