How to implement property hijacking with JavaScript defineProperty

How to implement property hijacking with JavaScript defineProperty

Preface

defineProperty 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.

Descriptors

Let 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:

  • configurable attribute: whether the descriptor can be modified, that is, whether other attributes of the descriptor can be modified again
  • enumerable property: whether the property can be enumerated, that is, whether the property can be for
  • writable attribute: whether the attribute value can be modified, that is, whether obj.a = 1 can be modified like this
  • value attribute: the value of the attribute
  • get attribute: It is a function. When the attribute is accessed, the function is automatically called, and the function return value is the value of the attribute.
  • set property: is a function. When the property is modified, the function is automatically called. The function has one and only one parameter, the new value to be assigned.

Notice! ! !

  • The value attribute, writable attribute and get attribute, set attribute in the descriptor are mutually exclusive. Only one can exist.
  • The default values ​​of other properties are all false. If you don’t want them to be false, remember to configure them. I won’t go into details (mainly because I don’t use them much).

Detailed explanation of get and set

  • get attribute: It is a function. When the attribute is accessed, the function is automatically called, and the function return value is the value of the attribute.
  • set property: is a function. When the property is modified, the function is automatically called. The function has one and only one parameter, the new value to be assigned.

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 object

With 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 object

Try 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 type

There 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

  • Unable to monitor object to add attributes
  • Unable to monitor object deletion attributes
  • Cannot hijack array modifications

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 properties

In 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 logs

For 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

Summarize

This 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

Recommend

Summary of MySQL database and table sharding

During project development, our database data is ...

RHCE installs Apache and accesses IP with a browser

1. at is configured to write "This is a at t...

JavaScript Shorthand Tips

Table of contents 1. Merge arrays 2. Merge arrays...

How to migrate sqlite to mysql script

Without further ado, I will post the code for you...

Sample code for cool breathing effect using CSS3+JavaScript

A simple cool effect achieved with CSS3 animation...

Use CSS to easily implement some frequently appearing weird buttons

background In the group, some students will ask r...

How to Learn Algorithmic Complexity with JavaScript

Table of contents Overview What is Big O notation...

Why the disk space is not released after deleting data in MySQL

Table of contents Problem Description Solution Pr...

IE6 BUG and fix is ​​a preventive strategy

Original article: Ultimate IE6 Cheatsheet: How To...

Solution to mysql server 5.5 connection failure

The solution to the problem that mysql cannot be ...

Operate on two columns of data as new columns in sql

As shown below: select a1,a2,a1+a2 a,a1*a2 b,a1*1...

css add scroll to div and hide the scroll bar

CSS adds scrolling to div and hides the scroll ba...