sequence

When you document data binding and responses in Vue, you note that you cannot update responses to objects that have passed through the object.freeze () method. Therefore, I specifically looked up the meaning of the object.freeze () method.

meaning

The object.freeze () method is used to freeze an Object and prevent changes to its properties. (Since arrays are objects by nature, this method can be used with arrays.) Mozilla MDN is described as follows:

You can freeze an object. A frozen object can no longer be modified. If you freeze an object, you cannot add new attributes to the object, delete existing attributes, modify the enumerability, configurability, writability of existing attributes of the object, or modify the value of existing attributes. In addition, the stereotype of an object cannot be modified after it is frozen

The return value of this method is its argument itself.

There are two points to note

  1. Object.freeze(), unlike const declarations, does not assume the function of const.

    Const is quite different from object.freeze ()

  • Const behaves like let. The only difference is that const defines a variable that cannot be reassigned. Variables declared by const are block-scoped, rather than function-scoped variables declared by var.
  • Object.freeze() takes an Object as an argument and returns the same immutable Object. This means that we cannot add, delete, or change any properties of the object.
  • Const is different from object.freeze () in that const prevents variable reassignment, whereas object.freeze () makes the Object immutable.

The following code is correct:

  1. Object.freeze() is a “shallow freeze” and the following code is in effect:

The instance

Regular use

It is obvious that the prop property of A has not been changed, even though it has been reassigned.

extension

“Deep freeze”

To completely freeze objects with nested properties, you can write your own library or use an existing library to freeze objects, such as Deepfreeze or immutation-js

// Deep freeze function.functionDeepFreeze (obj) {/ / retrieve the definition in the property name on the obj var propNames = Object. GetOwnPropertyNames (obj); // Freeze the property propnames.foreach (before freezing itselffunction(name) { var prop = obj[name]; // If prop is an object, freeze itif (typeof prop == 'object'&& prop ! == null) deepFreeze(prop); }); // Freeze itself (no-opif already frozen)
  return Object.freeze(obj);
}
Copy the code

It’s just a simple recursive method. But involves a very important, but rarely use of knowledge in writing business logic Object. GetOwnPropertyNames (obj). We all know that there are prototype chain attributes in JS Object, through this method can get all non-prototype chain attributes.

usingObject.freeze()Improve performance

In addition to component optimization, we can also start with the dependency transformation of VUE. At initialization, Vue does getter and setter modifications to data. In modern browsers, this process is actually quite fast, but there is still room for optimization.

Object.freeze() can freeze an Object. After freezing, you cannot add new attributes to the Object, modify the value of its existing attributes, delete existing attributes, and modify the enumerability, configurability, and writability of existing attributes of the Object. This method returns the frozen object.

When you pass a normal JavaScript Object to the Vue instance’s data option, Vue iterates through all of the Object’s properties and converts them into getters/setters using Object.defineProperty. These getters/setters are invisible to the user, but internally they let Vue track dependencies and notify changes when properties are accessed and modified.

But when Vue encounters an Object property that has been set to unconfigurable, like Object.freeze(), it does not apply data hijacking methods such as setters and getters to the Object. Refer to Vue source code

Vue observer source

Comparison of performance improvement effects

In a VUe-based Big Table benchmark, you can see how to render a 1000 x 10 table with Object.freeze() turned on before and after rerendering.

big table benchmark

Before tuning

After tuning

In this example, using object.freeze () is 4 times faster than not using it

whyObject.freeze()Performance will be better

Do not useObject.freeze()The CPU overhead

useObject.freeze()The CPU overhead

By contrast, using object.freeze () reduces the overhead of the observer.

Object.freeze()Application scenarios

Since object.freeze () freezes objects, it’s a good idea to show classes. If your data attributes need to change, you can replace them with a new object.freeze () Object.

Javascript object unfreezing

Modifying React props Objects generated by React cannot be modified. However, in practice, you need to modify the props. Object.isfrozen () ¶ if the props Object isFrozen, this is true. The properties of the object are read-only.

So, is there a way to unfreeze the PROPS object so that it can be modified?

In fact, in javascript, once an object is frozen, there is no way to unfreeze it. The only way to do this is to clone a new object with the same properties and modify the properties of the new object.

It can go like this:

ES6: Object.assign({}, frozenObject);
lodash: _.assign({}, frozenObject);
Copy the code

Look at the actual code:

function modifyProps(component) {
  let condictioin = this.props.condictioin,
    newComponent = Object.assign({}, component),
    newProps = Object.assign({}, component.props)
  
  if (condictioin) {
    if (condictioin.add) newProps.add = true
    if (condictioin.del) newProps.del = true
  }
  newComponent.props = newProps
  
  return newComponent
}
Copy the code

Method of locking an object

  • Object.preventExtensions()

No new properties or methods can be added to the project object

  • Object.seal()

Same as Prevent extension, plus prevents existing properties and methods from being deleted

  • Object.freeze()

Same as seal, plus prevent existing properties and methods from being modified On this basis, all properties of the object are read-only and cannot be modified

Object. IsSealed (), Object. IsFrozen (), and Object

Object.freeze() prevents Vue from implementing responsive systems

When a Vue instance is created, it adds all the attributes found in its data object to the Vue’s reactive system. When the values of these properties change, the view generates a “response,” that is, the match is updated to the new value. However, if you use Object.freeze(), this prevents modifying existing attributes, meaning that the response system can no longer track the changes.

Examples of specific use methods:

< template > < div > < p > after the freeze will change you {{obj. Foo}} < / p > <! -- Neither of them can be modified. Why is that? The second should theoretically be modifiable --> < button@click ="change"</button> </div> </template> <script> var obj = {foo:'Not going to change'
}
Object.freeze(obj)
export default {
  name: 'index'.data () {
    return {
      obj: obj
    }
  },
  methods: {
    change () {
      this.obj.foo = 'change'
    }
  }
}
</script>
Copy the code

After the operation:

The read-only attribute foo cannot be modified. Object.freeze() freezes the value. You can still replace the reference to the variable and change the above code to:

<button @click="change"</button>change () {
      this.obj = {
        foo: 'Will change'}}Copy the code

Object.freeze() is a new feature in ES5. You can freeze an Object. Freeze means that you cannot add new attributes to the Object, modify the value of existing attributes, delete existing attributes, and modify the enumerability, configurability, and writability of existing attributes of the Object. Prevents objects from being modified. If you have a large array or Object and are sure the data won’t change, using object.freeze () can give performance a big boost.

Practical experience and skills

Object.freeze() is a new feature in ES5 that freezes an Object to prevent it from being modified.

Vue 1.0.18+ supports this. For data or Vuex objects that are frozen using freeze, Vue does not convert getters to setters.

If you have a large array or Object and are sure the data won’t change, using object.freeze () can give performance a big boost. In my actual development, the increase is about 5 to 10 times, increasing with the amount of data.

Also, object.freeze () freezes values, and you can still replace references to variables. Here’s an example:

<p v-for="item in list">{{ item.value }}</p>
Copy the code
Freeze ([{value: 1}, {value: 2}])} new Vue({data: {// Vue will not do getter, setter binding for list object.created() {this.list[0]. Value = 100; This. list = [{value: 100}, {value: 200}]; this.list = Object.freeze([ { value: 100 }, { value: 200 } ]); }})Copy the code

This feature is not documented in the Vue documentation, but it is a very practical practice, and you can use Object.freeze to improve performance for purely presentable big data.