Implement a simple data response system

Implement a simple data response system

1. Dep

In fact, this is a container for dependency collection. depend collects dependencies, and notify triggers dependencies.

class Dep{
  constructor() {
    this._subs = [];
  }
  depend() {
    this._subs.push(Dep.target)
  }
  notify() {
    this._subs.forEach(item => {
      item.fn();
    })
  }
}

// Actually, this is the beginning of the love between dep and watcher. // A global property is used in watcher to store watcher
Dep.target = null;

function pushTarget(watch) {
  Dep.target = watch;
}

function popTarget() {
  Dep.target = null;
}

2. Understand obverser

Recursively convert all properties of the data object into accessor properties

// Convert to accessor property function defineReactive (obj, key, val, shallow) {

  // Create a dependency collection container let dep = new Dep();
  let childOb = !shallow && observe(val)

  Object.defineProperty(obj, key, {
    enumerable: true,
    configurable: true,
    get: function reactiveGetter () {
      if(Dep.target) {
        // Collect dependencies dep.depend();
      }
      return val;
      // ...
    },
    set: function reactiveSetter (newVal) {
      if(newVal === val) return;
      // Continue recursively traversing observe(newVal);
      //Trigger dependency dep.notify();
      // ...
    }
  })
}

class Observer{
  constructor(data) {
    this.walk(data);
  }

  walk(data) {
    const keys = Object.keys(data)
    for (let i = 0; i < keys.length; i++) {
      defineReactive(data, keys[i], data[keys[i]])
    }
  }
}

// Recursively convert all properties of the data object into accessor properties function observe (data) {
  if(Object.prototype.toString.call(data) !== '[object Object]') return;
  new Observer(data);
}

At this point, you can convert all properties of any object into accessors

3. Understand watch and observer

const data = {
  a: 1,
  b: 2
}

//First monitor an object observe(data);

The main function of watcher is to detect a certain attribute and trigger a callback when the attribute changes.

class Watcher{
  /**
  * @params {Function} exp a property expression * @params {Function} fn callback */
  constructor(exp, fn) {
    this.exp = exp;
    this.fn = fn;

    //Save watcher
    // Dep.target = this;
    pushTarget(this);

    // Execute the expression function once first, and in the calling process,
    // Trigger the accessor of data.a, and the get of data.a is executed.
    // Trigger dep.depend() to start collecting dependencies this.exp();

    // Release Dep.target
    popTarget();
  }
}

// new Watcher Such a dependency is collected new Watcher(() => {
  return data.a + data.b;
}, () => {
  console.log('change')
})

4. Trigger Dependency

data.a = 3; // change
data.b = 3; // change

5. Summarize the process

  • Convert all properties of an object into accessors
  • When adding watcher to a property, get function of the property will be triggered. The watcher will be stored in the dep dependency container of the property in the get function.
  • When this property changes, the set method of the property will be triggered. set function will execute all the dependencies stored in dep

This is the end of this article about implementing a simple data response system. For more relevant data response system content, please search for previous articles on 123WORDPRESS.COM or continue to browse the following related articles. I hope everyone will support 123WORDPRESS.COM in the future!

You may also be interested in:
  • Use of data response system for Vue source code analysis

<<:  Introduction to the use of common Dockerfile commands

>>:  How to control the proportion of Flex child elements on the main axis

Recommend

MySQL slow query optimization: the advantages of limit from theory and practice

Many times, we expect the query result to be at m...

Detailed explanation of the usage of MySQL data type DECIMAL

MySQL DECIMAL data type is used to store exact nu...

Introduction and use of five controllers in K8S

Table of contents Controller type of k8s Relation...

A simple method to implement scheduled backup of MySQL database in Linux

Here are the detailed steps: 1. Check the disk sp...

Setting up shared folders in Ubuntu virtual machine of VMWare14.0.0

This is my first blog post. Due to time constrain...

Problems and solutions of using jsx syntax in React-vscode

Problem Description After installing the plugin E...

Teach you how to implement Vue3 Reactivity

Table of contents Preface start A little thought ...

Linux kernel device driver advanced character device driver notes

/****************** * Advanced character device d...

Implementing the preview function of multiple image uploads based on HTML

I recently wrote a script for uploading multiple ...

Basic usage tutorial of MySQL slow query log

Slow query log related parameters MySQL slow quer...

TypeScript generic parameter default types and new strict compilation option

Table of contents Overview Create a type definiti...

Implementation of Docker packaging image and configuration modification

I have encountered many problems in learning Dock...

How to bind domain name to nginx service

Configure multiple servers in nginx.conf: When pr...

Zen HTML Elements Friends who use zen coding can collect it

html ¶ <html></html> html:xml ¶ <h...

Implementing user registration function with js

This article example shares the specific code of ...