v-model原理问题
v-model的原理
很多同学在理解Vue的时候都把Vue的数据响应原理理解为双向绑定,但实际上这是不准确的,我们之前提到的数据响应,都是通过数据的改变去驱动DOM重新的变化,而双向绑定已有数据驱动DOM外,DOM的变化反过来影响数据,是一个双向关系,在Vue中,我们可以通过v-model
来实现双向绑定。
v-model
即可以作用在普通表单元素上,又可以作用在组件上,它实际上是一个语法糖,接下来我们就来分析v-model
的实现原理。
表单元素
为了更加直观,我们还是结合示例来分析:
1 let vm = new Vue({
2 el: '#app',
3 template: '<div>'
4 + '<input v-model="message" placeholder="edit me">' +
5 '<p>Message is: {{ message }}</p>' +
6 '</div>',
7 data() {
8 return {
9 message: ''
10 }
11 }
12 })
这是一个非常简单的演示,我们在input
元素上设置了v-model
属性,绑定了message
,当我们在input
上输入了内容,message
也会同步变化。接下来我们就来分析Vue是如何实现这一效果的,其实非常简单。
也是先从编译阶段分析,首先是parse
阶段,v-model
被当做普通的指令解析到el.directives
中,然后在codegen
阶段,执行genData
的时候,会执行const dirs = genDirectives(el, state)
,它的定义在src/compiler/codegen/index.js
中:
1 function genDirectives (el: ASTElement, state: CodegenState): string | void {
2 const dirs = el.directives
3 if (!dirs) return
4 let res = 'directives:['
5 let hasRuntime = false
6 let i, l, dir, needRuntime
7 for (i = 0, l = dirs.length; i < l; i++) {
8 dir = dirs[i]
9 needRuntime = true
10 const gen: DirectiveFunction = state.directives[dir.name]
11 if (gen) {
12 // compile-time directive that manipulates AST.
13 // returns true if it also needs a runtime counterpart.
14 needRuntime = !!gen(el, dir, state.warn)
15 }
16 if (needRuntime) {
17 hasRuntime = true
18 res += `{name:"${dir.name}",rawName:"${dir.rawName}"${
19 dir.value ? `,value:(${dir.value}),expression:${JSON.stringify(dir.value)}` : ''
20 }${
21 dir.arg ? `,arg:"${dir.arg}"` : ''
22 }${
23 dir.modifiers ? `,modifiers:${JSON.stringify(dir.modifiers)}` : ''
24 }},`
25 }
26 }
27 if (hasRuntime) {
28 return res.slice(0, -1) + ']'
29 }
30 }
genDrirectives
方法就是遍历el.directives
,然后获取每一个指令对应的方法const gen: DirectiveFunction = state.directives[dir.name]
,这个指令方法实际上是在实例化CodegenState
的时候通过option
引用的,这个option
就是编译相关的配置,它在不同的平台下配置不同,在web
环境下的定义在src/platforms/web/compiler/options.js
下:
1 export const baseOptions: CompilerOptions = {
2 expectHTML: true,
3 modules,
4 directives,
5 isPreTag,
6 isUnaryTag,
7 mustUseProp,
8 canBeLeftOpenTag,
9 isReservedTag,
10 getTagNamespace,
11 staticKeys: genStaticKeys(modules)
12 }
13 directives定义在src/platforms/web/compiler/directives/index.js中:
14
15
16 export default {
17 model,
18 text,
19 html
20 }
21 // 那么对于v-model而言,对应的directive函数是在src/platforms/web/compiler/directives/model.js中定义的model函数:
22
23 export default function model (
24 el: ASTElement,
25 dir: ASTDirective,
26 _warn: Function
27 ): ?boolean {
28 warn = _warn
29 const value = dir.value
30 const modifiers = dir.modifiers
31 const tag = el.tag
32 const type = el.attrsMap.type
33
34 if (process.env.NODE_ENV !== 'production') {
35 // inputs with type="file" are read only and setting the input's
36 // value will throw an error.
37 if (tag === 'input' && type === 'file') {
38 warn(
39 `<${el.tag} v-model="${value}" type="file">:\n` +
40 `File inputs are read only. Use a v-on:change listener instead.`
41 )
42 }
43 }
44
45 if (el.component) {
46 genComponentModel(el, value, modifiers)
47 // component v-model doesn't need extra runtime
48 return false
49 } else if (tag === 'select') {
50 genSelect(el, value, modifiers)
51 } else if (tag === 'input' && type === 'checkbox') {
52 genCheckboxModel(el, value, modifiers)
53 } else if (tag === 'input' && type === 'radio') {
54 genRadioModel(el, value, modifiers)
55 } else if (tag === 'input' || tag === 'textarea') {
56 genDefaultModel(el, value, modifiers)
57 } else if (!config.isReservedTag(tag)) {
58 genComponentModel(el, value, modifiers)
59 // component v-model doesn't need extra runtime
60 return false
61 } else if (process.env.NODE_ENV !== 'production') {
62 warn(
63 `<${el.tag} v-model="${value}">: ` +
64 `v-model is not supported on this element type. ` +
65 'If you are working with contenteditable, it\'s recommended to ' +
66 'wrap a library dedicated for that purpose inside a custom component.'
67 )
68 }
69
70 // ensure runtime directive metadata
71 return true
72 }
也就是说我们执行needRuntime = !!gen(el, dir, state.warn)
就是在执行model
函数,它会根据AST元素例程的不同情况去执行不同的逻辑,对于我们这个案例而言,它会命中genDefaultModel(el, value, modifiers)
的逻辑,稍后我们也会介绍组件的处理,其他分开同学们可以自行去看。我们来看一下genDefaultModel
的实现:
1 function genDefaultModel (
2 el: ASTElement,
3 value: string,
4 modifiers: ?ASTModifiers
5 ): ?boolean {
6 const type = el.attrsMap.type
7
8 // warn if v-bind:value conflicts with v-model
9 // except for inputs with v-bind:type
10 if (process.env.NODE_ENV !== 'production') {
11 const value = el.attrsMap['v-bind:value'] || el.attrsMap[':value']
12 const typeBinding = el.attrsMap['v-bind:type'] || el.attrsMap[':type']
13 if (value && !typeBinding) {
14 const binding = el.attrsMap['v-bind:value'] ? 'v-bind:value' : ':value'
15 warn(
16 `${binding}="${value}" conflicts with v-model on the same element ` +
17 'because the latter already expands to a value binding internally'
18 )
19 }
20 }
21
22 const { lazy, number, trim } = modifiers || {}
23 const needCompositionGuard = !lazy && type !== 'range'
24 const event = lazy
25 ? 'change'
26 : type === 'range'
27 ? RANGE_TOKEN
28 : 'input'
29
30 let valueExpression = '$event.target.value'
31 if (trim) {
32 valueExpression = `$event.target.value.trim()`
33 }
34 if (number) {
35 valueExpression = `_n(${valueExpression})`
36 }
37
38 let code = genAssignmentCode(value, valueExpression)
39 if (needCompositionGuard) {
40 code = `if($event.target.composing)return;${code}`
41 }
42
43 addProp(el, 'value', `(${value})`)
44 addHandler(el, event, code, null, true)
45 if (trim || number) {
46 addHandler(el, 'blur', '$forceUpdate()')
47 }
48 }
genDefaultModel
函数先处理了modifiers
,它的不同主要影响的是event
和valueExpression
的值,对于我们的示例,event
为input
,valueExpression
为$event.target.value
。然后去执行genAssignmentCode
去生成代码,它的定义在src/compiler/directives/model.js
中:
1 /**
2 * Cross-platform codegen helper for generating v-model value assignment code.
3 */
4 export function genAssignmentCode (
5 value: string,
6 assignment: string
7 ): string {
8 const res = parseModel(value)
9 if (res.key === null) {
10 return `${value}=${assignment}`
11 } else {
12 return `$set(${res.exp}, ${res.key}, ${assignment})`
13 }
14 }
该方法首先对对应v-model
的精心解决value
,它处理了非常多的情况,对我们的例子,value
就是messgae
,所以返回的res.key
为null
,然后我们就得到${value}=${assignment}
,也就是message=$event.target.value
。然后我们又命中了needCompositionGuard
为true的逻辑,所以最终的code
为if($event.target.composing)return;message=$event.target.value
。
code
生成完后,又执行了2句非常关键的代码:
1 addProp(el, 'value', `(${value})`)
2 addHandler(el, event, code, null, true)
这实际上就是input
实现v-model
的精髓,通过修改AST元素,给el
添加一个prop
,相当于我们在input
上动态绑定了value
,又给el
添加了事件处理,相当于在input
上绑定了input
事件,实际上转换成模板如下:
1 <input
2 v-bind:value="message"
3 v-on:input="message=$event.target.value">
其实就是动态绑定了input
的value
指向messgae
变量,并且在触发input
事件的时候去动态把message
设置为目标值,这样实际上就完成了数据双向绑定了,所以说v-model
实际上就是语法糖。
再回到genDirectives
,它接下来的逻辑就是根据指令生成一些data
的代码:
1 if (needRuntime) {
2 hasRuntime = true
3 res += `{name:"${dir.name}",rawName:"${dir.rawName}"${
4 dir.value ? `,value:(${dir.value}),expression:${JSON.stringify(dir.value)}` : ''
5 }${
6 dir.arg ? `,arg:"${dir.arg}"` : ''
7 }${
8 dir.modifiers ? `,modifiers:${JSON.stringify(dir.modifiers)}` : ''
9 }},`
10 }
对我们的例子而言,最终生成的render
代码如下:
1 with(this) {
2 return _c('div',[_c('input',{
3 directives:[{
4 name:"model",
5 rawName:"v-model",
6 value:(message),
7 expression:"message"
8 }],
9 attrs:{"placeholder":"edit me"},
10 domProps:{"value":(message)},
11 on:{"input":function($event){
12 if($event.target.composing)
13 return;
14 message=$event.target.value
15 }}}),_c('p',[_v("Message is: "+_s(message))])
16 ])
17 }
关于事件的处理我们之前的章节已经分析过了,所以对于input
的v-model
而言,完全就是语法糖,并且对于其他表单元素套路都是一样,区别在于生成的事件代码会略有不同。
v-model
除了作用在表单元素上,新版的Vue还把这一语法糖用在了组件上,接下来我们来分析它的实现。
本文转自 http://caibaojian.com/vue-analysis/extend/v-model.html
v-model原理问题的更多相关文章
- DPM(Deformable Parts Model)--原理(一)(转载)
DPM(Deformable Parts Model) Reference: Object detection with discriminatively trained partbased mode ...
- DPM(Deformable Parts Model)--原理(一)
http://blog.csdn.net/ttransposition/article/details/12966521 DPM(Deformable Parts Model) Reference: ...
- miaov- 自动生成正V反V大于号V小于号V楼梯等图案
1. 核心:控制 数量的长度-1-i的位置,是放在left上还是top上?是放在前面还是后面! <!DOCTYPE html> <html lang="en"&g ...
- 使用RStudio调试(debug)基础学习(二)和fGarch包中的garchFit函数估计GARCH模型的原理和源码
一.garchFit函数的参数--------------------------------------------- algorithm a string parameter that deter ...
- 什么是V模型?使用SDLC和STLC学习案例研究
本教程详细介绍了软件/系统开发生命周期(SDLC),如瀑布循环和迭代循环,如RAID和Agile.此外,它继续解释测试的V模型和STLC(软件测试生命周期). 假设为您分配了一项任务,即为客户开发自定 ...
- 不可错过的效能利器「GitHub 热点速览 v.22.39」
如果你是一名前端工程师且维护着多个网站,不妨试试本周榜上有名的 HTML-first 的 Qwik,提升网站访问速度只用一招.除了提升网站加载速度的 Qwik,本周周榜上榜的 Whisper 也是一个 ...
- vue.js初级入门之最基础的双向绑定操作
首先在页面引入vue.js以及其他需要用到的或者可能要用到的插件(这里我多引用了bootstrap和jquery) 引用的时候需要注意文件的路径,准备工作这样基本就完成了,下面正式开始入门. vue. ...
- backbone学习总结(二)
今天来看下backbone的路由控制的功能.其实个人感觉backbone,模块就那么几个,熟悉它的框架结构,以及组成,就差不多. 废话不多说,我们来看看还剩下的功能. 关于路由和历史管理 通过 Bac ...
- backbone学习总结(一)
入职第三天,新公司项目用到backbone+underscore+require等框架,前两天把项目的开发环境都配置好啦,项目也能跑起来,现在准备好好学习公司自己的框架以及用到的框架,有点想吐槽,开发 ...
- Backbone学习总结
Backbone中文学习文档:http://www.css88.com/doc/backbone/ 来到公司已经有一段时间了,到现在深深的感觉到自己的能力弱的像只周黑鸭,又干涩又黝黑,充满了麻(手麻脑 ...
随机推荐
- Cubieboard安装系统
2013年买的一个小玩意. 一.硬件 1.1 相关资料 http://www.cubieforums.com http://cubie.cc 1.2 cubieboard1 1.3 无线网卡 水星 M ...
- undefined与null与?. ??
undefined: undefined是全局对象的一个属性,在一下情况下都是undefined: 当一个变量没有被赋值: 当一个函数没有返回值: 当某个对象不存在某个属性却去访问: 当函数定义了形参 ...
- MongoDB启动报错:Unrecognized option: storage try 'mongod --help' for more information(已解决)
问题说明: 今天在使用配置文件方式启动MongoDB时,一直启动失败,报错显示:Unrecognized option: storage try 'mongod --help' for more in ...
- application.properties文件中暗藏玄机
上次分享了如何一步一步搭建一个springboot的项目,详细参见<5分钟快速搭建一个springboot的项目>,最终的结果是在"8080"端口搭建起了服务,并成功访 ...
- 【Java面试】请说一下ReentrantLock的实现原理?
一个工作了3年的粉丝私信我,在面试的时候遇到了这样一个问题. "请说一下ReentrantLock的实现原理",他当时根据自己的理解零零散散的说了一些. 但是似乎没有说到关键点上, ...
- Linux常用命令-创建用户修改密码-useradd
命令简介 useradd/userdel 创建新用户/删除用户,需要管理员权限操作. 在创建用户时,如果不配置密码,用户的默认密码是不可用的,所以,useradd命令一般与passwd命令配合使用,下 ...
- Pytorch实现波阻抗反演
Pytorch实现波阻抗反演 1 引言 地震波阻抗反演是在勘探与开发期间进行储层预测的一项关键技术.地震波阻抗反演可消除子波影响,仅留下反射系数,再通过反射系数计算出能表征地层物性变化的物理参数.常用 ...
- npm发布包以及更新包还有需要注意的几点问题(这里以发布vue插件为例)
前言 在此之前,你需要去npm官网注册一个属于自己的账号,记住自己的账户名以及密码.邮箱,后面会用的到.第一步,安装webpack简易框架 vue init webpack-simple marque ...
- 记一次 .NET 某物管后台服务 卡死分析
一:背景 1. 讲故事 这几个月经常被朋友问,为什么不更新这个系列了,哈哈,确实停了好久,主要还是打基础去了,分析 dump 的能力不在于会灵活使用 windbg,而是对底层知识有一个深厚的理解,比如 ...
- kvm虚拟机在线扩容
fdisk -l查看当前虚拟机磁盘容量 1. 镜像扩容 先操作镜像,给镜像增加2T容量: 关闭虚拟机back_log,然后再宿主机上给虚拟机扩容 qemu-img info /home/kvm/bac ...