v-model实现原理是什么?v-model的使用方法介绍(附代码)

vue中vue-model原理是什么?v-model是vue用于表单元素上创建双向数据绑定,它本质是一个语法糖,在单向数据绑定的基础上,增加了监听用户输入事件并更新数据的功能。下面的文章中将给大家说一说v-model的实现原理及其使用方法的介绍。

v-model实现原理

genDirectives

Vue初始化组件时通过genDirectives(el,state)初始化指令。(这里的el已经通过parseHTML将html结构转换成Vue的AST语法树,state是根据用户定义组件的options新建的CodegenState对象)。

请输入:
你输入的是:

登录后复制

当用户在html页面上写了v-model指令进行数据双向绑定,Vue通过state找到model指令对应的方法model(el,dir,_warn),并执行该方法。(这里双!!表示将函数返回结果转换成Boolean类型)

var gen = state.directives[dir.name];    if (gen) {      needRuntime = !!gen(el, dir, state.warn);    }

登录后复制

model

if (el.component) {    genComponentModel(el, value, modifiers);    return false  } else if (tag === 'select') {    genSelect(el, value, modifiers);  } else if (tag === 'input' && type === 'checkbox') {    genCheckboxModel(el, value, modifiers);  } else if (tag === 'input' && type === 'radio') {    genRadioModel(el, value, modifiers);  } else if (tag === 'input' || tag === 'textarea') {    genDefaultModel(el, value, modifiers);  } else if (!config.isReservedTag(tag)) {    genComponentModel(el, value, modifiers);    return false  } else {    warn$1(      ": " +      "v-model is not supported on this element type. " +      'If you are working with contenteditable, it's recommended to ' +      'wrap a library dedicated for that purpose inside a custom component.'    );  }

登录后复制

model()根据表单元素的tag标签以及type属性的值,调用不同的方法也就验证了官网所说的“随表单控件类型不同而不同。”这里调用的就是genDefaultModel().

genDefaultModel

var type = el.attrsMap.type;

登录后复制

获取表单元素的类型,此处type=’text’.

{    var value$1 = el.attrsMap['v-bind:value'] || el.attrsMap[':value'];    var typeBinding = el.attrsMap['v-bind:type'] || el.attrsMap[':type'];    if (value$1 && !typeBinding) {      var binding = el.attrsMap['v-bind:value'] ? 'v-bind:value' : ':value';      warn$1(        binding + "="" + value$1 + "" conflicts with v-model on the same element " +        'because the latter already expands to a value binding internally'      );    }  }

登录后复制

检测该表单元素是否同时有v-model绑定和v-bind:value。

  var ref = modifiers || {};  var lazy = ref.lazy;  var number = ref.number;  var trim = ref.trim;  var needCompositionGuard = !lazy && type !== 'range';  var event = lazy    ? 'change'    : type === 'range'      ? RANGE_TOKEN      : 'input';

登录后复制

获取修饰符lazy,number及trim.

.lazy 取代input监听change事件

.number 输入字符串转为数字

.trim 输入首尾空格过滤

 var valueExpression = '$event.target.value';  if (trim) {    valueExpression = "$event.target.value.trim()";  }  if (number) {    valueExpression = "_n(" + valueExpression + ")";  }

登录后复制

定义变量valueExpression,本例子的情况valueExpression为’$event.target.value’。

var code = genAssignmentCode(value, valueExpression);

登录后复制

function genAssignmentCode (  value,  assignment) {  var res = parseModel(value);  if (res.key === null) {    return (value + "=" + assignment)  } else {    return ("$set(" + (res.exp) + ", " + (res.key) + ", " + assignment + ")")  }}

登录后复制

通过genAssignmentCode()方法生成v-model value值得代码。根据本文的例子返回的结果就是”message=$event.target.value”。

if (needCompositionGuard) {    code = "if($event.target.composing)return;" + code;  }

登录后复制

添加event.target.composing判断。event.target.composing用于判断此次input事件是否是IME构成触发的,如果是IME构成,直接return。IME 是输入法编辑器(Input Method Editor) 的英文缩写,IME构成指我们在输入文字时,处于未确认状态的文字。

带下划线的ceshi就属于IME构成,它会同样会触发input事件,但不会触发v-model更新数据。

addProp(el, 'value', ("(" + value + ")"));

登录后复制

function addProp (el, name, value) {  (el.props || (el.props = [])).push({ name: name, value: value });  el.plain = false;}

登录后复制

给el添加prop

2761451174-5b6037f864c59_articlex.png

addHandler(el, event, code, null, true);

登录后复制

function addHandler (el,name,value,modifiers,important,warn){    /*其他代码省略*/    var newHandler = {        value: value.trim()      };    var handlers = events[name];      if (Array.isArray(handlers)) {        important ? handlers.unshift(newHandler) : handlers.push(newHandler);      } else if (handlers) {        events[name] = important ? [newHandler, handlers] : [handlers, newHandler];      } else {        events[name] = newHandler;      }}

登录后复制

将code作为el的对应event处理的方法handler,此处events[‘input’]=if($event.target.composing)return;message =$event.target.value
最后原来的html结构就由:


登录后复制

变成了:


登录后复制

添加修饰符

如果添加了trim修饰符
原来的html结构就由:


登录后复制

变成了:


登录后复制

如果添加了lazy修饰符
原来的html结构就由:


登录后复制

变成了:


登录后复制

如果添加了number修饰符
原来的html结构就由:


登录后复制

变成了:


登录后复制

这里的_n是在installRenderHelpers里面定义的,指向toNumber方法。

function installRenderHelpers (target) {  /*其他代码省略*/  target._n = toNumber;}

登录后复制

而toNumber就是使用parseFloat函数来转的数字,再使用isNaN判断转换结果,如果结果是NaN,那么就返回原字符串,否则返回转为数字后的结果:

function toNumber (val) {  var n = parseFloat(val);  return isNaN(n) ? val : n}

登录后复制

注意:number修饰符不能限制输入的内容,就算输入的不是数字,也可能会被转换,如输入’1次测试’,转换的结果就是1

相关文章推荐:

vue中v-model动态使用详解

vue v-model表单控件绑定的实例教程

Vue 进阶教程之:v-model的详解

以上就是v-model实现原理是什么?v-model的使用方法介绍(附代码)的详细内容,更多请关注【创想鸟】其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至253000106@qq.com举报,一经查实,本站将立刻删除。

发布者:PHP中文网,转转请注明出处:https://www.chuangxiangniao.com/p/2744718.html

(0)
上一篇 2025年3月8日 03:35:43
下一篇 2025年3月8日 03:35:49

AD推荐 黄金广告位招租... 更多推荐

相关推荐

发表回复

登录后才能评论