使用JavaScript数组,我可以用一个赋值将其重置为空状态:

array.length = 0;

这使得数组“显示”为空,可以重用,据我所知,这是一个单一的“操作”——也就是说,常数时间。

有没有类似的方法来清除JS对象?我知道我可以迭代它的字段删除它们:

for (var prop in obj) { if (obj.hasOwnProperty(prop)) { delete obj[prop]; } }

但这有线性复杂度。

我也可以扔掉这个对象并创建一个新的对象:

obj = {};

但是“混杂”创建新对象会导致IE6上的垃圾收集问题。(如下所述)


当前回答

我认为,对你的问题的简短回答是否定的(你可以创建一个新对象)。

在本例中,我认为将长度设置为0仍然会将所有元素留给垃圾收集。 您可以将此添加到Object。如果它是你经常使用的东西,就创建原型。是的,它在复杂性上是线性的,但以后任何不进行垃圾收集的东西都会是线性的。 这是最好的解决办法。我知道这和你的问题无关,但是我们需要继续支持IE6多久呢?现在有很多运动要求停止使用它。

如果上面有什么不正确的地方,请随时纠正我。

其他回答

有一些新的东西可以让你去思考,去期待。在ES7和一般的数据绑定中观察。考虑:

var foo={
   name: "hello"
};

Object.observe(foo, function(){alert('modified');}); // bind to foo

foo={}; // You are no longer bound to foo but to an orphaned version of it
foo.name="there"; // This change will be missed by Object.observe()

所以在这种情况下,2号可能是最好的选择。

这困扰了我很久,所以这里是我的版本,因为我不想要一个空对象,我想要一个所有的属性,但重置为一些默认值。有点像类的新实例化。

let object1 = { a: 'somestring', b: 42, c: true, d:{ e:1, f:2, g:true, h:{ i:"hello" } }, j: [1,2,3], k: ["foo", "bar"], l:["foo",1,true], m:[{n:10, o:"food", p:true }, {n:11, o:"foog", p:true }], q:null, r:undefined }; let boolDefault = false; let stringDefault = ""; let numberDefault = 0; console.log(object1); //document.write("<pre>"); //document.write(JSON.stringify(object1)) //document.write("<hr />"); cleanObject(object1); console.log(object1); //document.write(JSON.stringify(object1)); //document.write("</pre>"); function cleanObject(o) { for (let [key, value] of Object.entries(o)) { let propType = typeof(o[key]); //console.log(key, value, propType); switch (propType) { case "number" : o[key] = numberDefault; break; case "string": o[key] = stringDefault; break; case "boolean": o[key] = boolDefault; break; case "undefined": o[key] = undefined; break; default: if(value === null) { continue; } cleanObject(o[key]); break; } } } // EXPECTED OUTPUT // Object { a: "somestring", b: 42, c: true, d: Object { e: 1, f: 2, g: true, h: Object { i: "hello" } }, j: Array [1, 2, 3], k: Array ["foo", "bar"], l: Array ["foo", 1, true], m: Array [Object { n: 10, o: "food", p: true }, Object { n: 11, o: "foog", p: true }], q: null, r: undefined } // Object { a: "", b: 0, c: undefined, d: Object { e: 0, f: 0, g: undefined, h: Object { i: "" } }, j: Array [0, 0, 0], k: Array ["", ""], l: Array ["", 0, undefined], m: Array [Object { n: 0, o: "", p: undefined }, Object { n: 0, o: "", p: undefined }], q: null, r: undefined }

简单的方法:

Object.keys(object).forEach(key => {
  delete object[key];
})

对象。键会以数组的形式返回对象的每个键,例如:

const user = {
  name: 'John Doe',
  age: 25,
};

Object.keys(user) // ['name', 'age']

forEach将在每个元素的第一个参数中运行函数,而delete关键字将从对象中删除一个键。 因此,代码将为每个键删除对象的一个键。

我认为,对你的问题的简短回答是否定的(你可以创建一个新对象)。

在本例中,我认为将长度设置为0仍然会将所有元素留给垃圾收集。 您可以将此添加到Object。如果它是你经常使用的东西,就创建原型。是的,它在复杂性上是线性的,但以后任何不进行垃圾收集的东西都会是线性的。 这是最好的解决办法。我知道这和你的问题无关,但是我们需要继续支持IE6多久呢?现在有很多运动要求停止使用它。

如果上面有什么不正确的地方,请随时纠正我。

ES5

ES5解决方案可以是:

// for enumerable and non-enumerable properties
Object.getOwnPropertyNames(obj).forEach(function (prop) {
  delete obj[prop];
});

ES6

而ES6解决方案可以是:

// for enumerable and non-enumerable properties
for (const prop of Object.getOwnPropertyNames(obj)) {
  delete obj[prop];
}

性能

不管规格如何,最快的解决方案通常是:

// for enumerable and non-enumerable of an object with proto chain
var props = Object.getOwnPropertyNames(obj);
for (var i = 0; i < props.length; i++) {
  delete obj[props[i]];
}

// for enumerable properties of shallow/plain object
for (var key in obj) {
  // this check can be safely omitted in modern JS engines
  // if (obj.hasOwnProperty(key))
    delete obj[key];
}

for. in应该只在浅对象或普通对象上执行的原因是,它遍历原型继承的属性,而不仅仅是自己的可以删除的属性。如果不确定对象是普通的且属性是可枚举的,则使用object。getOwnPropertyNames是一个更好的选择。