Inhaltsverzeichnis
Grundkenntnisse
Object.defineProperty
Publish-Subscribe-Muster
Praktische Praxis
Fazit
Heim Web-Frontend js-Tutorial Detaillierte Erläuterung des Reaktionsfähigkeitsprinzips von vue.j (mit Code)

Detaillierte Erläuterung des Reaktionsfähigkeitsprinzips von vue.j (mit Code)

Nov 16, 2018 pm 04:52 PM
html5 javascript vue.js 前端

Dieser Artikel bietet Ihnen eine detaillierte Erklärung des Reaktionsfähigkeitsprinzips von vue.js (mit Code). Ich hoffe, dass er für Sie hilfreich ist.

Ich bin schon vor langer Zeit mit AngularJS in Berührung gekommen. Damals habe ich bereits verstanden, dass AngularJS die Datenüberwachung und das Rendern von Seitenaktualisierungen durch Dirty Checking implementiert. Danach kam ich mit vue.js in Kontakt. Damals war ich sehr neugierig, wie vue.js Datenaktualisierungen überwacht und die Seite neu rendert. Heute werden wir das Prinzip der Reaktionsfähigkeit von vue.j Schritt für Schritt analysieren und eine einfache Demo implementieren.

Lassen Sie uns zunächst einige Grundlagen verstehen.

Grundkenntnisse

Object.defineProperty

es5 hat die API Object.defineProperty hinzugefügt, mit der wir Getter und Setter für die Eigenschaften von Objekten festlegen können, damit wir es können die Werterfassung und Zuweisung von Objekteigenschaften durch den Benutzer kapern. Zum Beispiel der folgende Code:

const obj = {
};

let val = 'cjg';
Object.defineProperty(obj, 'name', {
  get() {
    console.log('劫持了你的取值操作啦');
    return val;
  },
  set(newVal) {
    console.log('劫持了你的赋值操作啦');
    val = newVal;
  }
});

console.log(obj.name);
obj.name = 'cwc';
console.log(obj.name);
Nach dem Login kopieren

Wir haben die Werterfassungs- und Zuweisungsoperationen von obj[name] über Object.defineProperty gekapert, sodass wir hier einige Tricks anwenden können. Beispielsweise können wir obj[name] verwenden. Löst bei Zuweisung den Seitenaktualisierungsvorgang aus.

Publish-Subscribe-Muster

Publish-Subscribe-Muster ist eines der gebräuchlichsten Designmuster, bei dem es zwei Rollen gibt: Herausgeber und Abonnent. Mehrere Abonnenten können ein Ereignis desselben Herausgebers abonnieren. Wenn das Ereignis eintritt, benachrichtigt der Herausgeber alle Abonnenten, die das Ereignis abonniert haben. Schauen wir uns zum Verständnis ein Beispiel an.

class Dep {
  constructor() {
    this.subs = [];
  }
  // 增加订阅者
  addSub(sub) {
    if (this.subs.indexOf(sub) < 0) {
      this.subs.push(sub);
    }
  }
  // 通知订阅者
  notify() {
    this.subs.forEach((sub) => {
      sub.update();
    })
  }
}

const dep = new Dep();

const sub = {
  update() {
    console.log('sub1 update')
  }
}

const sub1 = {
  update() {
    console.log('sub2 update');
  }
}

dep.addSub(sub);
dep.addSub(sub1);
dep.notify(); // 通知订阅者事件发生,触发他们的更新函数
Nach dem Login kopieren

Praktische Praxis

Nachdem wir Object.defineProperty und das Publish-Subscriber-Modell verstanden haben, können wir uns leicht vorstellen, dass vue.js die Datenüberwachung basierend auf den beiden oben genannten implementiert.

  1. vue.js verwendet zunächst Object.defineProperty, um die Getter und Setter der zu überwachenden Daten zu kapern. Wenn die Eigenschaften der Daten zugewiesen/abgerufen werden, kann vue.js erkennen und entsprechend behandeln.

  2. Durch das Abonnementveröffentlichungsmodell können wir für jedes Attribut des Objekts einen Herausgeber erstellen. Wenn andere Abonnenten von diesem Attribut abhängig sind, wird der Abonnent zur Veröffentlichung hinzugefügt . Mithilfe der Datenübernahme von Object.defineProperty benachrichtigt der Herausgeber der Eigenschaft alle Abonnenten über den aktualisierten Inhalt, wenn der Setter der Eigenschaft aufgerufen wird.

Als nächstes implementieren wir es (siehe Kommentare für Details):

class Observer {
  constructor(data) {
    // 如果不是对象,则返回
    if (!data || typeof data !== 'object') {
      return;
    }
    this.data = data;
    this.walk();
  }

  // 对传入的数据进行数据劫持
  walk() {
    for (let key in this.data) {
      this.defineReactive(this.data, key, this.data[key]);
    }
  }
  // 创建当前属性的一个发布实例,使用Object.defineProperty来对当前属性进行数据劫持。
  defineReactive(obj, key, val) {
    // 创建当前属性的发布者
    const dep = new Dep();
    /*
    * 递归对子属性的值进行数据劫持,比如说对以下数据
    * let data = {
    *   name: 'cjg',
    *   obj: {
    *     name: 'zht',
    *     age: 22,
    *     obj: {
    *       name: 'cjg',
    *       age: 22,
    *     }
    *   },
    * };
    * 我们先对data最外层的name和obj进行数据劫持,之后再对obj对象的子属性obj.name,obj.age, obj.obj进行数据劫持,层层递归下去,直到所有的数据都完成了数据劫持工作。
    */
    new Observer(val);
    Object.defineProperty(obj, key, {
      get() {
        // 若当前有对该属性的依赖项,则将其加入到发布者的订阅者队列里
        if (Dep.target) {
          dep.addSub(Dep.target);
        }
        return val;
      },
      set(newVal) {
        if (val === newVal) {
          return;
        }
        val = newVal;
        new Observer(newVal);
        dep.notify();
      }
    })
  }
}

// 发布者,将依赖该属性的watcher都加入subs数组,当该属性改变的时候,则调用所有依赖该属性的watcher的更新函数,触发更新。
class Dep {
  constructor() {
    this.subs = [];
  }

  addSub(sub) {
    if (this.subs.indexOf(sub) < 0) {
      this.subs.push(sub);
    }
  }

  notify() {
    this.subs.forEach((sub) => {
      sub.update();
    })
  }
}

Dep.target = null;

// 观察者
class Watcher {
  /**
   *Creates an instance of Watcher.
   * @param {*} vm
   * @param {*} keys
   * @param {*} updateCb
   * @memberof Watcher
   */
  constructor(vm, keys, updateCb) {
    this.vm = vm;
    this.keys = keys;
    this.updateCb = updateCb;
    this.value = null;
    this.get();
  }

  // 根据vm和keys获取到最新的观察值
  get() {
    Dep.target = this;
    const keys = this.keys.split('.');
    let value = this.vm;
    keys.forEach(_key => {
      value = value[_key];
    });
    this.value = value;
    Dep.target = null;
    return this.value;
  }

  update() {
    const oldValue = this.value;
    const newValue = this.get();
    if (oldValue !== newValue) {
      this.updateCb(oldValue, newValue);
    }
  }
}

let data = {
  name: 'cjg',
  obj: {
    name: 'zht',
  },
};

new Observer(data);
// 监听data对象的name属性,当data.name发现变化的时候,触发cb函数
new Watcher(data, 'name', (oldValue, newValue) => {
  console.log(oldValue, newValue);
})

data.name = 'zht';

// 监听data对象的obj.name属性,当data.obj.name发现变化的时候,触发cb函数
new Watcher(data, 'obj.name', (oldValue, newValue) => {
  console.log(oldValue, newValue);
})

data.obj.name = 'cwc';
data.obj.name = 'dmh';
Nach dem Login kopieren

Fazit

Auf diese Weise ist eine einfache responsive Datenüberwachung erledigt . Dies ist natürlich nur eine einfache Demo zur Veranschaulichung des Reaktionsfähigkeitsprinzips von vue.js. Der eigentliche Quellcode von vue.js wird komplizierter sein, da viele andere Logiken hinzugefügt werden.

Das obige ist der detaillierte Inhalt vonDetaillierte Erläuterung des Reaktionsfähigkeitsprinzips von vue.j (mit Code). Für weitere Informationen folgen Sie bitte anderen verwandten Artikeln auf der PHP chinesischen Website!

Erklärung dieser Website
Der Inhalt dieses Artikels wird freiwillig von Internetnutzern beigesteuert und das Urheberrecht liegt beim ursprünglichen Autor. Diese Website übernimmt keine entsprechende rechtliche Verantwortung. Wenn Sie Inhalte finden, bei denen der Verdacht eines Plagiats oder einer Rechtsverletzung besteht, wenden Sie sich bitte an admin@php.cn

Heiße KI -Werkzeuge

Undresser.AI Undress

Undresser.AI Undress

KI-gestützte App zum Erstellen realistischer Aktfotos

AI Clothes Remover

AI Clothes Remover

Online-KI-Tool zum Entfernen von Kleidung aus Fotos.

Undress AI Tool

Undress AI Tool

Ausziehbilder kostenlos

Clothoff.io

Clothoff.io

KI-Kleiderentferner

Video Face Swap

Video Face Swap

Tauschen Sie Gesichter in jedem Video mühelos mit unserem völlig kostenlosen KI-Gesichtstausch-Tool aus!

Heiße Werkzeuge

Notepad++7.3.1

Notepad++7.3.1

Einfach zu bedienender und kostenloser Code-Editor

SublimeText3 chinesische Version

SublimeText3 chinesische Version

Chinesische Version, sehr einfach zu bedienen

Senden Sie Studio 13.0.1

Senden Sie Studio 13.0.1

Leistungsstarke integrierte PHP-Entwicklungsumgebung

Dreamweaver CS6

Dreamweaver CS6

Visuelle Webentwicklungstools

SublimeText3 Mac-Version

SublimeText3 Mac-Version

Codebearbeitungssoftware auf Gottesniveau (SublimeText3)

Tabellenrahmen in HTML Tabellenrahmen in HTML Sep 04, 2024 pm 04:49 PM

Anleitung zum Tabellenrahmen in HTML. Hier besprechen wir verschiedene Möglichkeiten zum Definieren von Tabellenrändern anhand von Beispielen für den Tabellenrand in HTML.

Verschachtelte Tabelle in HTML Verschachtelte Tabelle in HTML Sep 04, 2024 pm 04:49 PM

Dies ist eine Anleitung für verschachtelte Tabellen in HTML. Hier diskutieren wir anhand der entsprechenden Beispiele, wie man eine Tabelle innerhalb der Tabelle erstellt.

HTML-Rand links HTML-Rand links Sep 04, 2024 pm 04:48 PM

Anleitung zum HTML-Rand links. Hier besprechen wir einen kurzen Überblick über HTML margin-left und seine Beispiele sowie seine Code-Implementierung.

HTML-Tabellenlayout HTML-Tabellenlayout Sep 04, 2024 pm 04:54 PM

Leitfaden zum HTML-Tabellenlayout. Hier besprechen wir die Werte des HTML-Tabellenlayouts zusammen mit den Beispielen und Ausgaben im Detail.

HTML-Eingabeplatzhalter HTML-Eingabeplatzhalter Sep 04, 2024 pm 04:54 PM

Leitfaden für HTML-Eingabeplatzhalter. Hier besprechen wir die Beispiele für HTML-Eingabeplatzhalter zusammen mit den Codes und Ausgaben.

Text in HTML verschieben Text in HTML verschieben Sep 04, 2024 pm 04:45 PM

Anleitung zum Verschieben von Text in HTML. Hier besprechen wir eine Einführung, wie Marquee-Tags funktionieren, mit Syntax und Beispielen für die Implementierung.

HTML-geordnete Liste HTML-geordnete Liste Sep 04, 2024 pm 04:43 PM

Leitfaden zur HTML-geordneten Liste. Hier besprechen wir auch die Einführung von HTML-geordneten Listen und Typen sowie deren Beispiele

HTML-Onclick-Button HTML-Onclick-Button Sep 04, 2024 pm 04:49 PM

Anleitung zum HTML-OnClick-Button. Hier diskutieren wir deren Einführung, Funktionsweise, Beispiele und Onclick-Events in verschiedenen Veranstaltungen.

See all articles