PrefacedefineProperty is the core of Vue to achieve data hijacking. This article explains how defineProperty achieves property hijacking bit by bit. In fact, we generally operate object properties in the way that we add or modify properties, and we can use Object.defineProperty. let obj = {}; // Common operation: add/modify new attribute obj.a = 1; // is equivalent to: Object.defineProperty(o, "a", { value: 1, writable: true, configurable: true, enumerable: true }); Of course, we wouldn't play it this way in ordinary examples, as it would be too long-winded. But defineProperty can be more precise to add or modify the properties of an object. DescriptorsLet me start with a proper noun: descriptor. In fact, it is the third parameter of defineProperty, which is an object. This object has the following properties:
Notice! ! !
Detailed explanation of get and set
Repeat silently three times and memorize. Write an example of get and set to help understand. This example must be mastered. After understanding it, you will basically grasp the essence of data hijacking. let obj = {}; let value = 1; Object.defineProperty(obj, "b", { get() { console.log("Read b attribute", value); return value; }, set(newValue) { console.log("Set b property", newValue); value = newValue; } }); // Trigger the get function, the return value of get is the attribute value // 1 console.log(obj.b); // Trigger the set function, the value becomes 2, note! ! ! At this time, the attribute value in the memory has not changed obj.b = 2; // However, when you want to read the property value, the get function will inevitably be triggered, and the property value will naturally change. This idea is really good console.log(obj.b); There is a pitfall here: there can be no read operations in get, otherwise it will be an endless loop, so where get set is used, a variable is always needed So, here, the value of the variable value is the value of the attribute. If you want to modify the attribute, just modify the value of value. I think that's enough to understand the essence of get and set after understanding this example. Hijacking an attribute of an objectWith the foundation of the previous example, try to write any property of the hijacked object. function observeKey(obj, key) { let value = obj[key]; Object.defineProperty(obj, key, { get() { console.log("Read properties", value); return value; }, set(newValue) { console.log("Set properties", newValue); value = newValue; } }); } let obj = { a: 1 }; observeKey(obj, "a"); // Read a, trigger the get function console.log(obj.a); // Set a, trigger the set function obj.a = 1; Hijack all properties of an objectTry to hijack all the properties of the object In fact, it is traversal: function observeObj(obj) { for (let key in obj) { // Directly using obj.hasOwnProperty will prompt non-standard if (Object.prototype.hasOwnProperty.call(obj, key)) { observeKey(obj, key); } } return obj; } function observeKey(obj, key) { let value = obj[key]; Object.defineProperty(obj, key, { get() { console.log("Read properties", value); return value; }, set(newValue) { console.log("Set properties", newValue); value = newValue; } }); } let obj = { a: 1, b: 2 }; observeObj(obj); console.log(obj); // Read a, trigger the get function console.log(obj.a); // Set a, trigger the set function obj.a = 1; Hijack all properties of an object - including the property value of the object typeThere is a flaw in the above, that is, when the attribute value is also an object, the attribute value cannot be hijacked, such as {a:1,c:{b:1}} Simple, recursion, just fill it in. function observeObj(obj) { // Add parameter restrictions, only objects can be hijacked, which is also the termination condition of recursion if (typeof obj !== "object" || obj == null) { return; } for (let key in obj) { // Directly using obj.hasOwnProperty will prompt non-standard if (Object.prototype.hasOwnProperty.call(obj, key)) { observeKey(obj, key); // Here, the attribute value of the attribute is hijacked. If it is not an object, it is returned directly without affecting observeObj(obj[key]); } } return obj; } function observeKey(obj, key) { let value = obj[key]; Object.defineProperty(obj, key, { get() { console.log("Read properties", value); return value; }, set(newValue) { console.log("Set properties", newValue); value = newValue; } }); } let obj = { a: 1, b: 2, c: { name: "c" } }; observeObj(obj); console.log(obj); // Read a, trigger the get function console.log(obj.a); // Set a, trigger the set function obj.a = 1; // Trigger set function obj.c.name = "d"; Note that the observeObj function cannot hijack new properties of an object, but can only hijack existing properties of the object. The drawbacks of defineProperty
Of course, array modifications can be monitored in other ways, which is achieved by hijacking the array change method. The above defects are also why there are $set/$delete in Vue and why arrays can only be detected using specific methods. let obj = { a: 1, b: [1, 2] }; observeObj(obj); // Add new attributes obj.c = 3; // Will not trigger the get function console.log(obj.c); // Will not trigger the set function obj.b.push(3); defineProperty can also mount propertiesIn fact, it is to access options.data.name, which can be abbreviated as options.name, a professional term, to mount the attributes on data to options This is equivalent to using defineProperty to add new properties to options: // Mount a single attribute first // options.data is equivalent to source options is equivalent to target function proxyKey(target, source, key) { Object.defineProperty(target, key, { // Here source[key] is equivalent to the variable value, so the simplest example is the core get() { return source[key]; }, set(newValue) { if (newValue === source[key]) { return; } source[key] = newValue; } }); } // Traverse the attributes and mount function proxyObj(target, source) { for (let key in source) { // Directly using obj.hasOwnProperty will prompt non-standard if (Object.prototype.hasOwnProperty.call(source, key)) { proxyKey(target, source, key); } } } let options = { data: { name: 1 } }; proxyObj(options, options.data); // 1 console.log(options.name); By the way, the core principles of Vue's attribute hijacking and mounting attributes are almost the same as above. defineProperty can also write logsFor example, obj has an attribute whose value changes frequently, and we want to record all its changed values to form a log. let obj = { a: 1 }; let log = [obj.a]; let value = obj.a; Object.defineProperty(obj, "a", { get() { return value; }, set(newValue) { if (newValue === value) { return; } value = newValue; log.push(newValue); } }); obj.a = 2; obj.a = 3; obj.a = 4; // [1,2,3,4] console.log(log); A general class can be extracted to record the changes of a certain value. class Archiver { constructor() { let value = null; this.archive = []; Object.defineProperty(this, "a", { get() { return value; }, set(newValue) { if (newValue === value) { return; } value = newValue; this.archive.push(newValue); } }); } } let archiver = new Archiver(); archiver.a = 1; archiver.a = 2; // [1,2] console.log(archiver.archive); References defineProperty from MDN SummarizeThis is the end of this article about how to implement property hijacking with JavaScript defineProperty. For more related defineProperty property hijacking content, please search 123WORDPRESS.COM’s previous articles or continue to browse the following related articles. I hope everyone will support 123WORDPRESS.COM in the future! |
<<: Ubuntu Docker installation in vmware (container building)
>>: Mysql solves the database N+1 query problem
During project development, our database data is ...
1. at is configured to write "This is a at t...
Table of contents 1. Merge arrays 2. Merge arrays...
Without further ado, I will post the code for you...
A simple cool effect achieved with CSS3 animation...
background In the group, some students will ask r...
Preface It's a cliché. Here I will talk about...
Table of contents Overview What is Big O notation...
Table of contents Problem Description Solution Pr...
With the continuous development of the Internet ec...
Original article: Ultimate IE6 Cheatsheet: How To...
Table of contents 1. CentOS7+MySQL8.0, yum source...
The solution to the problem that mysql cannot be ...
As shown below: select a1,a2,a1+a2 a,a1*a2 b,a1*1...
CSS adds scrolling to div and hides the scroll ba...