Learn from video course summarized below: www.bilibili.com/video/BV1Zy…
First, Vue2 response principle
DefineProperty descriptor (ObJ, prop, descriptor)
- Parameter — obj: Object for which attributes are to be defined. Prop: The name or Symbol of the property to be defined or modified. Descriptor: Property descriptor to define or modify.
- Return value – the object passed to the function.
Attributes defined this way have the following advantages:
1. Finer control over the properties of defined objects
Enumerable, Writable, and any property that defaults to different is true.
var obj={ name:"rectangle", width:10, } object.defineProperty (obj,'height',{value:20}) // Enumerable for (let key in obj){console.log("@",key) } // can be modified writable obj.width=20; // The width can be deleted without any additional information, and the height can be modified without any additional information. //width deleted, height deleted failedCopy the code
Enumerable, writable, and any property defined by Object.defineProperty() is false by default, and can be changed without any additional information. Properties defined as follows can be traversed, modified, and deleted as literal properties.
Object.defineProperty(obj,'height',{
value:20,
enumerable:true,
writable:true,
configurable:true
})
Copy the code
2. Configure getters and setters for properties so that properties can be bound to specific variables.
-
Getter: This function is called when a property is accessed. This method returns a value that was obtained by accessing the property.
-
Setter: This function is called when the property value is modified. This method takes one parameter, which is the modified value of the property.
let number=30; Var rectangle ={name:”rectangle”, width:10,} Object.defineproperty (obj,’height’,{get:function(){console.log(“height attribute is read “); return number; }, set:function(value){console.log(“height attribute set “); number=value } }) obj.height; Obj. Height =22; //obj.height and number change to 22
(2) What is data broker
Data proxy: Operations (read/write) on properties in another object through an object proxy.
The following code is an example of obj2 proxy obj1’s x property operation.
let obj={x:100}; let obj2={y:200}; DefineProperty (obj2,'x',{get:function(){return obj.x; }, set:function(value){ obj.x=value } })Copy the code
(3) Data proxy of Vue2
Vue data proxy: The operation (read/write) of the attributes in the data object (Vue2:_data attribute, Vue3:$data attribute) through the Vue instance object (VM). The purpose of a data broker is to facilitate data access.
Add all attributes from the data Object to the VM via Object.defineProperty();
2. Specify a getter/setter for each attribute added to the VM. .
3. Operate (read/write) the corresponding property in the getter/setter inner section of data.
Ii. Vue2 Data Monitoring (Responsive-type principle)
Study the content of data where properties change and the content of the page changes.
(I) How to monitor data changes in Vue
Intercepts property reads and modifications (getter and setter methods) via Object.defineProperty(). Vue monitors the properties of objects at all levels of data through setters, that is, whether an object is an element of an array or a property of an object, its properties are monitored. Here is a simple example of implementing data monitoring.
Let data={name:"Alice", age:20} Function Observer(obj){for(let key in obj){object.defineProperty (this,key,{get(){return obj[key] }, set(val){ obj[key]=val } }) } } const obs=new Observer(data); console.log(obs); let vm={}; vm._data=data=obs;Copy the code
However, Object.defineProperty() intercepts data, and the following two scenarios do not.
- There are new attributes and deleted attributes.
- Modify the array directly by subscript, the interface does not automatically update;
(2) how to add responsive attributes to the object (monitored)?
There is no way to append attributes to objects in data directly after a Vue instance is created (there are no setter methods). Want to add attributes to do after the response, you need to use: Vue. Set (target, propertyName, value) or
vm.$set(target,propertyName,value)
const vm = new Vue({ el: '#app', data(){return {student:{name:'Alice', age:20}}}, methods:{//new Vue() AddSignature (){this.student.signature="I like summer"}, //new Vue() after creating the Vue instance, Vue. Set (this.student,"desc","I like winter")}} template:` <div> {{student.name}}-{{student.age}} <div v-if="student.signature">{{student.signature}}</div> <div><button </button></div> <div><button @click="addDesc" </button></div> </div> '})Copy the code
(c) if the property of the object is array, how to modify the array to be monitored?
To be monitored in Vue, you must use the following methods:
- Use apis that change arrays: push(), pop(), Shift (), unshift(), splice(), sort(), reverse(). Essentially, Vue calls the methods corresponding to the native array to update the array, reparse the template, and update the page.
- Vue. Set (target,index,value) or vm.$set(target,index,value) methods
We know that arrays are monitored as properties of objects, but we don’t monitor each item of an array value by adding a setter.
const vm = new Vue({ el: '#app', data(){ return { student:{ friends:[ {id:0,name:"Cindy"}, {id:1,name:"Mike"} ], hobby:['Singing','Swiming'], }}}})Copy the code
The console enters vm.student, which outputs the following:
You can see that Friends and Hobby are monitored as data object properties (setter methods), but Hobby [0] or Hobby [1] is not monitored, and friends[0] and Friends [1] are not monitored. But friends[0]/ Friends [1] are objects whose attributes ID and name are monitored. This means that
Friends [0]={id:2,name:"May"} // This way to modify the array is not allowed
Friends [0]. Name ="John" // This modification array is accepted
(4) Summary of Vue2 response formula principle
Object types: Intercepts reading and modifying attributes via Object.defineProperty() (data hijacking);
Array types: Intercepts by overriding some of the array’s column methods (wrapping the array’s change methods).
Existing problems:
- The interface does not update automatically when you add or delete attributes. Solution: vue.set () and vue.delete () methods.
- Modify the array directly by subscript, the interface does not automatically update. Solution: vue.set () or the method of changing the array.
Neither of these problems exists in VU3. This is essentially because Vue3 uses Proxy objects instead of Object.defineProperty() for data interception.
Vue3 Data Monitoring (Responsive principle)
(a) Proxy constructor
Create an object proxy that intercepts any property changes to the object, including reading, writing, adding, and deleting properties.
const p = new Proxy(target, handler)
Copy the code
- Target The target object (which can be any type of object, including a native array, a function, or even another Proxy) to be wrapped with a Proxy. (Proxy object)
- Handler An object, usually with functions as attributes, that define the behavior of agent P when performing various operations.
Official documentation: developer.mozilla.org/zh-CN/docs/…
Examples of using Proxy to Proxy objects are as follows. You can add, delete, change and check the properties of Proxy objects. You can also modify array elements by array subscript (P.sid[0]=9 successfully changed the array).
Rect ={name:"rectangle", width:10, side:[10,20]} /*p rect={name:"rectangle", width:10, side:[10,20]} Adding get, set, and deletePropert methods to the handler intercepts changes to rect properties and changes the page. Get (target,property){console.log(' read ${property} attribute 'on p') return rect[property]; }, // set(target,property,value){console.log(' changed ${property} 'on p); Rect [property]=value}, React deleteProperty(target,property){console.log(' delete ${property} property 'on p) return delete target[property]; }})Copy the code
Compared to Vue2 data interception, as shown below, directly deleting object attributes or adding object attributes does not take effect. Even if the 64x is configured with a different :true, only the CX on P is deleted, and the CX on RECT remains unchanged.
let rect={ name:"rectangle", width:10, } let p={}; Object.defineproperty (p,"width",{get(){// Call get() when accessing attributes (p.width). Console. log(' reads width from p ') return rect.width; }, // set(value){console.log(' changed width on p ') rect.width=value; }})Copy the code
(2) Reflect built-in object
Built-in objects that provide methods for intercepting objects that are part of the language and are the same as the proxy Handlers methods.
Obj ={a:1,b:2} reflect. get(obj,"a") // Returns 1 reflect. set(obj,"a",11) // Reflect.deleteProperty(obj,"b") // Delete b attribute successfullyCopy the code
Use Reflect and Proxy to Proxy the object as follows
Let p=new Proxy(rect,{get(target,property){console.log(' read ${property} attribute 'on p); return Reflect.get(rect,property); //return rect[property]; }, // set(target,property,value){console.log(' changed or added ${property} property 'on p); Reflect.set(rect,property,value); //rect[property]=value}, P and react change deleteProperty(target,property){console.log(' delete ${property} from p ') //return delete target[property]; return Reflect.deleteProperty(target,property) } })Copy the code