vue est un framework facile à utiliser avec de nombreuses fonctions pratiques intégrées à l'intérieur. La caractéristique la plus distinctive est son système réactif caché en bas. Les états des composants sont tous des objets JavaScript réactifs. Lorsque vous les modifiez, la vue est mise à jour, ce qui rend la gestion des états plus facile et plus intuitive. Alors, comment le système réactif Vue est-il implémenté ? Cet article est également basé sur la compréhension et l'implémentation de l'imitation après avoir lu le code source de Vue, alors suivez les idées de l'auteur et explorons Vue du moins profond au plus profond ! [Recommandations associées : Tutoriel vidéo vuejs]
La version du code source Vue de cet article : 2.6.14 Afin de faciliter la compréhension, le code est simplifié.
Lorsque vous transmettez un objet JavaScript ordinaire dans une instance Vue en tant qu'option de données, Vue parcourra toutes les propriétés de cet objet et utilisera Object.defineProperty pour convertir toutes ces propriétés en getters. /setters, puis parcourez les getters/setters.
Pour résumer le système réactif de Vue en une phrase : Mode observateur + getter/setter d'interception Object.defineProperty
Object.defineProperty()
Object.defineProperty()
方法会直接在一个对象上定义一个新属性,或者修改一个对象的现有属性,并返回此对象。
简单的说,就是通过此方式定义的 property,执行 obj.xxx
时会触发 get,执行 obj.xxx = xxx
会触发 set,这便是响应式的关键。
Object.defineProperty 是 ES5 中一个无法 shim(无法通过polyfill实现) 的特性,这也就是 Vue 不支持 IE8 以及更低版本浏览器的原因。
现在,我们来基于Object.defineProperty
définira directement une nouvelle propriété sur un objet, ou modifiera une propriété existante d'un objet, et renverra cet objet.
En termes simples, la propriété définie de cette manière déclenchera get lors de l'exécution de obj.xxx
, et set sera déclenchée lors de l'exécution de obj.xxx = xxx
. est la clé.
Implémentation de base d'un système réactif
Maintenant, implémentons un système de mise à jour réactif simple basé sur Object.defineProperty
comme « apéritif » let data = {};
// 使用一个中间变量保存 value
let value = "hello";
// 用一个集合保存数据的响应更新函数
let fnSet = new Set();
// 在 data 上定义 text 属性
Object.defineProperty(data, "text", {
enumerable: true,
configurable: true,
set(newValue) {
value = newValue;
// 数据变化
fnSet.forEach((fn) => fn());
},
get() {
fnSet.add(fn);
return value;
},
});
// 将 data.text 渲染到页面上
function fn() {
document.body.innerText = data.text;
}
// 执行函数,触发读取 get
fn();
// 一秒后改变数据,触发 set 更新
setTimeout(() => {
data.text = "world";
}, 1000);
Grâce au code ci-dessus, je pense que vous avez déjà une certaine compréhension du principe de fonctionnement du système réactif. Afin de rendre cet « apéritif » facile à digérer, ce système réactif simple présente encore de nombreux inconvénients : les fonctions de mise à jour des données et des réponses sont fortement couplées via un codage en dur, n'obtiennent qu'une situation un-à-un, et non modulaire. ça suffit. Attendez... Alors ensuite, complétons-le un par un.
Concevoir un système réactif completPour concevoir un système réactif complet, nous devons d'abord comprendre une connaissance pré-requise, quel est le modèle d'observateur ? Qu'est-ce que le modèle d'observateur ?
Il s'agit d'un modèle de conception comportemental qui vous permet de définir un mécanisme d'abonnement qui peut notifier plusieurs autres objets qui "observent" l'objet lorsqu'un événement d'objet se produit. Les objets avec des états remarquables sont souvent appeléscibles
Comme il doit avertir d'autres objets lorsque son propre état change, nous l'appelons également unéditeur
. Tous les autres objets qui souhaitent suivre les changements d'état de l'éditeur sont appelésabonnés
. De plus, l’éditeur interagit directement avec tous les abonnés uniquement via l’interface, et ils doivent tous avoir la même interface.Par exemple ? :
🎜Vous (c'est-à-dire l'abonné dans l'application) êtes intéressé par l'hebdomadaire d'une certaine librairie Vous laissez un numéro de téléphone à votre patron (c'est-à-dire l'éditeur dans l'application) et vous le laissez. sachez dès qu'il aura de nouvelles informations. L'hebdomadaire vous appellera, et les autres personnes intéressées par cet hebdomadaire laisseront également leur numéro de téléphone au patron. Lorsque le nouvel hebdomadaire arrive, le patron appelle un à un pour prévenir les lecteurs de venir le chercher. 🎜🎜Si un lecteur laisse accidentellement un numéro QQ au lieu d'un numéro de téléphone, l'ancienne version ne pourra pas passer lors de l'appel et le lecteur ne recevra pas de notifications. C’est ce que nous disions plus haut, il doit avoir la même interface. 🎜🎜Après avoir compris le modèle d'observateur, nous avons commencé à concevoir un système réactif. 🎜🎜🎜🎜Observateur abstrait (abonné) classe Watcher🎜🎜🎜在上面的例子中,数据和响应更新函数是通过硬编码强耦合在一起的。而实际开发过程中,更新函数不一定叫fn
,更有可能是一个匿名函数。所以我们需要抽像一个观察者(订阅者)类Watcher
来保存并执行更新函数,同时向外提供一个update
更新接口。
// Watcher 观察者可能有 n 个,我们为了区分它们,保证唯一性,增加一个 uid let watcherId = 0; // 当前活跃的 Watcher let activeWatcher = null; class Watcher { constructor(cb) { this.uid = watcherId++; // 更新函数 this.cb = cb; // 保存 watcher 订阅的所有数据 this.deps = []; // 初始化时执行更新函数 this.get(); } // 求值函数 get() { // 调用更新函数时,将 activeWatcher 指向当前 watcher activeWatcher = this; this.cb(); // 调用完重置 activeWatcher = null; } // 数据更新时,调用该函数重新求值 update() { this.get(); } }
抽象被观察者(发布者)类Dep
我们再想一想,实际开发过程中,data 中肯定不止一个数据,而且每个数据,都有不同的订阅者,所以说我们还需要抽象一个被观察者(发布者)Dep
类来保存数据对应的观察者(Watcher
),以及数据变化时通知观察者更新。
class Dep { constructor() { // 保存所有该依赖项的订阅者 this.subs = []; } addSubs() { // 将 activeWatcher 作为订阅者,放到 subs 中 // 防止重复订阅 if(this.subs.indexOf(activeWatcher) === -1){ this.subs.push(activeWatcher); } } notify() { // 先保存旧的依赖,便于下面遍历通知更新 const deps = this.subs.slice() // 每次更新前,清除上一次收集的依赖,下次执行时,重新收集 this.subs.length = 0; deps.forEach((watcher) => { watcher.update(); }); } }
抽象 Observer
现在,Watcher
和Dep
只是两个独立的模块,我们怎么把它们关联起来呢?
答案就是Object.defineProperty
,在数据被读取,触发get
方法,Dep 将当前触发 get 的 Watcher 当做订阅者放到 subs中,Watcher
就与 Dep
建立关系;在数据被修改,触发set
方法,Dep
就遍历 subs 中的订阅者,通知Watcher
更新。
下面我们就来完善将数据转换为getter/setter的处理。
上面基础的响应式系统实现中,我们只定义了一个响应式数据,当 data 中有其他property时我们就处理不了了。所以,我们需要抽象一个 Observer
类来完成对 data数据的遍历,并调用defineReactive
转换为 getter/setter,最终完成响应式绑定。
为了简化,我们只处理data中单层数据。
class Observer { constructor(value) { this.value = value; this.walk(value); } // 遍历 keys,转换为 getter/setter walk(obj) { const keys = Object.keys(obj); for (let i = 0; i <p>这里我们通过参数 value 的闭包,来保存最新的数据,避免新增其他变量</p><pre class="brush:php;toolbar:false">function defineReactive(target, key, value) { // 每一个数据都是一个被观察者 const dep = new Dep(); Object.defineProperty(target, key, { enumerable: true, configurable: true, // 执行 data.xxx 时 get 触发,进行依赖收集,watcher 订阅 dep get() { if (activeWatcher) { // 订阅 dep.addSubs(activeWatcher); } return value; }, // 执行 data.xxx = xxx 时 set 触发,遍历订阅了该 dep 的 watchers, // 调用 watcher.updata 更新 set(newValue) { // 如果前后值相等,没必要跟新 if (value === newVal) { return; } value = newValue; // 派发更新 dep.notify(); }, }); }
至此,响应式系统就大功告成了!!
测试
我们通过下面代码测试一下:
let data = { name: "张三", age: 18, address: "成都", }; // 模拟 render const render1 = () => { console.warn("-------------watcher1--------------"); console.log("The name value is", data.name); console.log("The age value is", data.age); console.log("The address value is", data.address); }; const render2 = () => { console.warn("-------------watcher2--------------"); console.log("The name value is", data.name); console.log("The age value is", data.age); }; // 先将 data 转换成响应式 new Observer(data); // 实例观察者 new Watcher(render1); new Watcher(render2);
在浏览器中运行这段代码,和我们期望的一样,两个render
都执行了,并且在控制台上打印了结果。
我们尝试修改 data.name = '李四 23333333'
,测试两个 render
都会重新执行:
我们只修改 data.address = '北京'
,测试一下是否只有render 1
回调都会重新执行:
都完美通过测试!!?
总结
Vue
响应式原理的核心就是Observer
、Dep
、Watcher
,三者共同构成 MVVM 中的 VM
Observer
中进行数据响应式处理以及最终的Watcher
和Dep
关系绑定,在数据被读的时候,触发get
方法,将 Watcher
收集到 Dep
中作为依赖;在数据被修改的时候,触发set
方法,Dep
就遍历 subs 中的订阅者,通知Watcher
更新。
本篇文章属于入门篇,并非源码实现,在源码的基础上简化了很多内容,能够便于理解Observer
、Dep
、Watcher
三者的作用和关系。
本文的源码,以及作者学习 Vue 源码完整的逐行注释源码地址:github.com/yue1123/vue…
Ce qui précède est le contenu détaillé de. pour plus d'informations, suivez d'autres articles connexes sur le site Web de PHP en chinois!