最新消息:雨落星辰是一个专注网站SEO优化、网站SEO诊断、搜索引擎研究、网络营销推广、网站策划运营及站长类的自媒体原创博客

Is there a good way to freeze an array of objects in Javascript? - Stack Overflow

programmeradmin2浏览0评论

So I have a Javascript module that looks like the following:

const data = [
     {
          id: 'do not modify',
          name: 'do not modify'
     },         
     {
          id: 'do not modify 2',
          name: 'do not modify 2'
     }
];

export default data;

Is there a clean way I can recursively freeze all objects in an array without explicitly calling Object.freeze() on each and every object? I realize I could just loop through the array and freeze each of them before exporting, but I was curious to know if there was a more elegant solution.

So I have a Javascript module that looks like the following:

const data = [
     {
          id: 'do not modify',
          name: 'do not modify'
     },         
     {
          id: 'do not modify 2',
          name: 'do not modify 2'
     }
];

export default data;

Is there a clean way I can recursively freeze all objects in an array without explicitly calling Object.freeze() on each and every object? I realize I could just loop through the array and freeze each of them before exporting, but I was curious to know if there was a more elegant solution.

Share Improve this question asked Aug 18, 2016 at 21:09 kibowkikibowki 4,37616 gold badges51 silver badges79 bronze badges 3
  • You can just freeze the array, which itself is an object – Glen Despaux Jr Commented Aug 18, 2016 at 21:11
  • 4 data.forEach(obj => Object.freeze(obj)); doesn't seem plicated to me. – zzzzBov Commented Aug 18, 2016 at 21:11
  • @zzzzBov: it's not plicated once you know of the Object.freeze() function, which I hadn't until just now (thanks!) :) – David Thomas Commented Aug 18, 2016 at 21:15
Add a ment  | 

4 Answers 4

Reset to default 9

All you'd have to do is pass Object.freeze to Array.prototype.forEach:

'use strict';
var objs = [
  { a: 1 },
  { b: 2 },
  { c: 3 }
];

objs.forEach(Object.freeze);
objs[0].a = 4; // Fails due to being frozen

data.forEach(Object.freeze) is the best you can do.

In your case, you might be able to clean up your code like so:

const data = [
  unmodifiableObj(0, 'Foo'),
  unmodifiableObj(1, 'Bar')
]

export default data;


function unmodifiableObj(id, name) {
  return Object.freeze({ id, name })
}

You can't freeze array with elements inside without calling freeze on every object. Here you can find example of deep freeze https://developer.mozilla/pl/docs/Web/JavaScript/Reference/Global_Objects/Object/freeze

I'm pasting code from page linked above:

obj1 = {
  internal: {}
};

Object.freeze(obj1);
obj1.internal.a = 'aValue';

obj1.internal.a // 'aValue'

// To make obj fully immutable, freeze each object in obj.
// To do so, we use this function.
function deepFreeze(obj) {

  // Retrieve the property names defined on obj
  var propNames = Object.getOwnPropertyNames(obj);

  // Freeze properties before freezing self
  propNames.forEach(function(name) {
    var prop = obj[name];

    // Freeze prop if it is an object
    if (typeof prop == 'object' && prop !== null)
      deepFreeze(prop);
  });

  // Freeze self (no-op if already frozen)
  return Object.freeze(obj);
}

obj2 = {
  internal: {}
};

deepFreeze(obj2);
obj2.internal.a = 'anotherValue';
obj2.internal.a; // undefined

Updated with sample code from MDN

Try this:

// To make obj fully immutable, freeze each object in obj.
// To do so, we use this function.
function deepFreeze(obj) {

// Retrieve the property names defined on obj
var propNames = Object.getOwnPropertyNames(obj);

// Freeze properties before freezing self
propNames.forEach(function(name) {
var prop = obj[name];

// Freeze prop if it is an object
if (typeof prop == 'object' && prop !== null)
  deepFreeze(prop);
});

// Freeze self (no-op if already frozen)
return Object.freeze(obj);
}

Which will freeze the array itself, as well as any objects it contains.

发布评论

评论列表(0)

  1. 暂无评论