1. v-model原理
vue中v-model是一个语法糖,所谓的语法糖就是对其他基础功能的二次封装而产生的功能。简单点说,v-model本身就是父组件对子组件状态以及状态改变事件的封装。其实现原理上分为两个部分:
通过props设置子组件的状态
通过监听子组件发出的事件改变父组件的状态,从而影响子组件的props值
通过以上两个部分,实现了父组件的状态和子组件状态进行了绑定的效果。
1.1 demo
v-model使用示例
<!DOCTYPE html> <html> <head> <meta charset="utf-8" /> <title>v-model示例</title> <script type="text/javascript" src="vue.js"></script> </head> <body> <div id="app"> <div>这里是父组件的状态:</div> <div style="margin-bottom: 15px;">{{content}}</div> <Child v-model="content"></Child> </div> <template id="input"> <div> <div>这里是子组件的输入区域:</div> <input :value="value" @input="contentChange" /> </div> </template> <script type="text/javascript"> var Child = { template: "#input", props: { value: { type: String, required: true } }, methods: { contentChange(value){ this.$emit("input", value.target.value); } } }; var vueInstance = new Vue({ el: "#app", components: {Child}, data: { content: "" } }) </script> </body> </html>
在浏览器中打开上述html页面,可以看到实时效果:在子组件中的input框中输入内容可以在父组件区域实时显示,达到了子组件中状态和父组件状态实时绑定的效果。
2. 修改v-model默认监听的事件和设置prop的名称
v-model指令默认是在子组件上设置的prop名称是value,默认监听子组件上的input事件,在上面的demo上,如果我们修改子组件contentChange函数中发出的事件名称,在父组件中就无法实时获取到子组件的输入。
Vue中提供了通过在子组件上定义model属性来修改这两个参数名称的功能,不过该功能需要在版本2.2以上才能使用,如下demo所示:
2.1 demo
<!DOCTYPE html> <html> <head> <meta charset="utf-8" /> <title>v-model示例</title> <script type="text/javascript" src="vue.js"></script> </head> <body> <div id="app"> <div>这里是父组件的状态:</div> <div style="margin-bottom: 15px;">{{content}}</div> <Child v-model="content"></Child> </div> <template id="input"> <div> <div>这里是子组件的输入区域:</div> <input :value="content" @input="contentChange" /> </div> </template> <script type="text/javascript"> var Child = { template: "#input", model: { prop: "content", event: "contentChanged" }, props: { content: { type: String, required: true } }, methods: { contentChange(value){ this.$emit("contentChanged", value.target.value); } } }; var vueInstance = new Vue({ el: "#app", components: {Child}, data: { content: "" } }) </script> </body> </html>
3. Vue中对v-model指令处理分析
基于Vue2.0版本,分析我们在标签上写上v-model属性到vue组件实现响应的流程。
3.1 解析部分
3.1.1 在将HTML解析称AST时,会解析HTML中标签的属性
function processAttrs(el){ ... name = name.replace(dirRE, '') // parse arg const argMatch = name.match(argRE) if (argMatch && (arg = argMatch[1])) { name = name.slice(0, -(arg.length + 1)) } addDirective(el, name, value, arg, modifiers) ... }
提取指令的名称,v-model的指令名称name为model,然后添加到实例的指令中
3.1.2 将指令相关内容添加到实例指令中
export function addDirective ( el: ASTElement, name: string, value: string, arg: ?string, modifiers: ?{ [key: string]: true } ) { (el.directives || (el.directives = [])).push({ name, value, arg, modifiers }) }
在实例的指令属性中添加相应的指令,这样就实现了从html上的属性到Vue实例上指令格式的转换
3.2 指令设置部分
在将html解析称AST之后,实例对应的directives属性上就有了我们设置的v-model相关的值,包括参数值value,name是model
3.2.1 调用指令的构造函数
function genDirectives (el: ASTElement): string | void { const dirs = el.directives if (!dirs) return let res = 'directives:[' let hasRuntime = false let i, l, dir, needRuntime for (i = 0, l = dirs.length; i < l; i++) { dir = dirs[i] needRuntime = true const gen = platformDirectives[dir.name] || baseDirectives[dir.name] if (gen) { // compile-time directive that manipulates AST. // returns true if it also needs a runtime counterpart. needRuntime = !!gen(el, dir, warn) } ... }
在v-model指令的构造函数中会根据tag的种类进行不同的创建函数进行创建,如果我们自定义指令需要在子组件上添加属性,也需要在这个函数里面进行操作
3.2.2 普通tag下的v-model指令构造过程
function genDefaultModel el: ASTElement, value: string, modifiers: ?Object ): ?boolean { ... addProp(el, 'value', isNative ? `_s(${value})` : `(${value})`) addHandler(el, event, code, null, true) ... }
3.3 指令响应变化部分
3.3.1 createPatchFunction统一处理指令的钩子函数
createPatchFunction函数返回一个patch函数,在patch处理过程中,会调用指令的钩子函数,包括:
4. 总结
4.1 编译过程
4.2 初始化过程
通过在patch函数中,调用统一的钩子函数,触发指令的钩子函数,实现相应的功能
以上就是详解vue v-model的详细内容,更多关于vue v-model的资料请关注小牛知识库其它相关文章!
本文向大家介绍awk 详解。相关面试题,主要包含被问及awk 详解。时的应答技巧和注意事项,需要的朋友参考一下 答案: awk '{pattern + action}' { filenames } #cat /etc/passwd |awk -F ':' '{print 1"t"7}' //-F 的意思是以':'分隔 root /bin/bash daemon /bin/sh 搜索/etc/pas
glob 是由普通字符和/或通配字符组成的字符串,用于匹配文件路径。可以利用一个或多个 glob 在文件系统中定位文件。 src() 方法接受一个 glob 字符串或由多个 glob 字符串组成的数组作为参数,用于确定哪些文件需要被操作。glob 或 glob 数组必须至少匹配到一个匹配项,否则 src() 将报错。当使用 glob 数组时,将按照每个 glob 在数组中的位置依次执行匹配 - 这
scanf()函数详解 #define _CRT_SECURE_NO_WARNINGS #include <stdio.h> #include <stdlib.h> //01.scanf();函数扫描输入事项: // 格式必须一一匹配:非格式控制符的可见字符必须一一匹配输入 int main01(void) { int num = 0; printf("%p \n", &nu
printf()详解 #define _CRT_SECURE_NO_WARNINGS #include <stdio.h> #include <stdlib.h> //01.printf();&sprintf();&fprintf();格式控制字符串详解: // (1).格式控制字符串的组成: // 普通字符串+格式控制符 // (2).常见的格式控制字符: // %f-
主要内容:state状态,自定义资源共享方式,源码实现AQS是AbstractQueuedSynchronizer的简称。AQS提供了一种实现阻塞锁和一系列依赖FIFO等待队列的同步器的框架,如下图所示。AQS为一系列同步器依赖于一个单独的原子变量(state)的同步器提供了一个非常有用的基础。子类们必须定义改变state变量的protected方法,这些方法定义了state是如何被获取或释放的。鉴于此,本类中的其他方法执行所有的排队和阻塞机制。子类
主要内容:state状态,自定义资源共享方式,源码实现AQS是AbstractQueuedSynchronizer的简称。AQS提供了一种实现阻塞锁和一系列依赖FIFO等待队列的同步器的框架,如下图所示。AQS为一系列同步器依赖于一个单独的原子变量(state)的同步器提供了一个非常有用的基础。子类们必须定义改变state变量的protected方法,这些方法定义了state是如何被获取或释放的。鉴于此,本类中的其他方法执行所有的排队和阻塞机制。子类
主要内容:6 DataNode(面试开发重点)6 DataNode(面试开发重点) 6.1 DataNode工作机制 DataNode工作机制,如图3-15所示。 1)一个数据块在DataNode上以文件形式存储在磁盘上,包括两个文件,一个是数据本身,一个是元数据包括数据块的长度,块数据的校验和,以及时间戳。 2)DataNode启动后向NameNode注册,通过后,周期性(1小时)的向NameNode上报所有的块信息。 3)心跳是每3秒一次
主要内容:4 HDFS的数据流,5 NameNode和SecondaryNameNode(面试开发重点)4 HDFS的数据流 4.1 HDFS写数据流程 4.1.1 剖析文件写入 HDFS写数据流程 1)客户端通过Distributed FileSystem模块向NameNode请求上传文件,NameNode检查目标文件是否已存在,父目录是否存在。 2)NameNode返回是否可以上传。 3)客户端请求第一个 Block上传到哪几个DataNode服务器上。 4)NameNode返回3个Data