vue3实现v-model原理详解
vue3 源码正式放出来了,想必大家也都开始争先恐后的学习 vue3 的知识了。由于 vue3 已经不再支持 v-model 了,而使用 .sync 来代替,为了这篇文章可以帮助大家快速了解 vue 的双向绑定实现原理,部分使用了 vue2.x v-model 的实现原理
proxy 的基础知识,相信大家已经都很了解了,让我们一起来回顾一下吧
proxy 是对一个对象的代理,并返回一个已代理的对象,已代理的对象如果发生任何 set 跟 get 的方法都可以被捕获到,我们写一个简单的 :chestnut:
const target = { a: 1 } const handers = { get() { // 当对 observed.a 进行取值时会触发 }, set() { // 当对 observed.a 进行赋值时会触发 }, // 还有一些额外的参数如 has 等,这里用不到,就不多说了 .... } const observed = new Proxy(target, handers)
这样我们就可以对 target 对象设置了一层代理,当我们对 target 进行取赋值操作的时候就可以接可以截获到它的行为了,如果你以为就只有这么简单你就错了。
我们把 target 改写成多层嵌套
const target = { a: { b: 1 } } ... const observed = new Proxy(target, handers)
我们再获取 observed.a.b = 2 的时候,get 方法取到的是 a 的值 { b: 1 }, 而 set 并不会触发。这也说明了 proxy 只能代理一层对象,不能深层代理!
那么我们需要监听到嵌套的对象怎么办?
其实这个也不难,就是在 get 的时候判断一下得到的值是不是对象,如果是对象的话就 在对它代理一层,直到一层,全部代理完为止,这里就需要一个递归函数
const target = { a: { b: 1 } } function reactive(data: any) { const handers = { get(target, key, receiver) { const res = Reflect.get(target, key, receiver); if (isObject(res)) { data[key] = reactive(res); } return target[key]; } } const observed = new Proxy(target, handers) }
这样我们就可以对目标函数内部的所有属性进行深层监听了,这样还是不够,因为我们每次取值的时候都会设置代理这样会导致代码无限循环->死循环,所以我们需要做一层判断,如果已经设置了代理的或这已经是代理的对象就不需要在此设置代理了。又因为我们要储存对象的映射,所以需要使用map函数。狼蚁网站SEO优化是reactive完整的代码。
const rawToReactive: WeakMap<any, any> = new WeakMap(); const reactiveToRaw: WeakMap<any, any> = new WeakMap(); function reactive(data: any) { // 已经有代理 let observed = rawToReactive.get(data); if (observed !== void 0) { return observed; } // 这个数据已经是代理 if (reactiveToRaw.has(data)) { return data; } const handler = { get: function(target: any, key: string, receiver: any) { const res = Reflect.get(target, key, receiver); if (isObject(res)) { data[key] = data[key] = reactive(res); } return target[key]; }, set: function(target: any, key: string, value: any) { // 将新值赋值 target[key] = value; // 通知所有订阅者触发更新 trigger(target); // 严格模式下需要设置返回值,否则会报错 return value; } }; // 返回代理监听对象 observed = new Proxy(data, handler as any); rawToReactive.set(data, observed); reactiveToRaw.set(observed, data); return observed; }
定义watcher 用来作为 pile 跟 reactive 的桥梁, 跟 vue3 的实现可能不一样
// 收集watcher依赖 const Dep: Dep = { deps: [], add(watcher: Watcher) { this.deps.push(watcher); } }; // observer跟pile的桥梁,在编译时添加watcher,在数据更新时触发update更新视图 function _watcher(node: any, attr: string, data: any, key: string): Watcher { return { node, attr, data, key, update() { // 逐层取值 const mutationKeys = this.key.split('.'); if (mutationKeys.length > 1) { let d: any = null; mutationKeys.forEach(key => (d = this.data[key] || (d && d[key]))); this.node[this.attr] = d; return; } this.node[this.attr] = this.data[this.key]; } }; }
接下来是编译模板
这里只是模拟编译,真正的编译不是这样的
获取到模板上的 v-model 、 v-bind 属性,获取到绑定的属性。当数据发生变化时,更新视图(这里会在trigger进行触发),当视图改变数据时修改数据(为了简单,通过eval函数实现),具体代码如下
// 编译模板 function _pile(nodes: any, $data: any) { [...nodes].forEach((e, index) => { const theNode = nodes[index]; // 获取到 input标签下的 v-model 属性,并添加watcher if (theNode.tagName === 'INPUT' && theNode.hasAttribute('v-model')) { const key = theNode.getAttribute('v-model'); Dep.add(_watcher(theNode, 'value', $data, key)); // 监听input事件 theNode.addEventListener('input', () => { const mutationKeys = key.split('.'); if (mutationKeys.length > 1) { eval(`$data.${key}='${theNode.value}'`); return; } $data[key] = theNode.value; }); } // 获取 v-bind 属性,并添加watcher if (theNode.hasAttribute('v-bind')) { const key = theNode.getAttribute('v-bind'); Dep.add(_watcher(theNode, 'innerHTML', $data, key)); } }); trigger($data); }
trigger 对依赖进行触发
function trigger(target: any, key?: string | symbol) { Dep.deps.forEach((e: Watcher) => { e.update(); }); }
使用效果
废话不多说。直接上代码!
假设我们有一个模板是这样的,接下来我们在这个模板的 id="my-app" 元素内实现双向绑定
<div id="my-app"> <h1 v-bind="a"></h1> <input v-model="a" type="text"> </div>
vue3 中 new Vue 已经被 createApp 所代替,reactive 是反应原理,可以抽出来单独使用,vue3 外漏了所有内部的 api,都可以在外部使用
const { createApp, reactive } = require('./vue.ts').default; const App = { setup() { const react = reactive({ a: { b: { c: { d: { e: 111 } } } } }); // 测试异步反应 setTimeout(() => { react.a.b.c.d.e = 222; }, 100); return react; } }; createApp().mount(App, '#my-app');
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持狼蚁SEO。
编程语言
- 如何快速学会编程 如何快速学会ug编程
- 免费学编程的app 推荐12个免费学编程的好网站
- 电脑怎么编程:电脑怎么编程网咯游戏菜单图标
- 如何写代码新手教学 如何写代码新手教学手机
- 基础编程入门教程视频 基础编程入门教程视频华
- 编程演示:编程演示浦丰投针过程
- 乐高编程加盟 乐高积木编程加盟
- 跟我学plc编程 plc编程自学入门视频教程
- ug编程成航林总 ug编程实战视频
- 孩子学编程的好处和坏处
- 初学者学编程该从哪里开始 新手学编程从哪里入
- 慢走丝编程 慢走丝编程难学吗
- 国内十强少儿编程机构 中国少儿编程机构十强有
- 成人计算机速成培训班 成人计算机速成培训班办
- 孩子学编程网上课程哪家好 儿童学编程比较好的
- 代码编程教学入门软件 代码编程教程