Vue中的this.$options.data()和this.$data用法说明


Posted in Javascript onJuly 26, 2020

问题

项目里遇到一个问题,用this.$options.data()重置组件data时,data()里用this获取的props或method都为undefined,代码简化如下:

export default {
  props: {
    P: Object
  },
  data () {
    return {
      A: {
        a: this.methodA
      },
      B: this.P
    };
  },
  methods: {
    resetData () { // 更新时调用
      Object.assign(this.$data, this.$options.data()); // 有问题!!!
    },
    methodA () {
      // do sth.
    },
    methodB () { // 通过用户操作调用
      this.A.a && this.A.a(); // this.A.a is undefined, this.B is undefined!!!
    }
  }
}

调用resetData()之后,再调用methodB()时,this.A.a和this.B是undefined。

解决

resetData里这样写:

resetData () { // 更新时调用
  Object.assign(this.$data, this.$options.data.call(this));
}

原因

和Vue实例的初始化相关。(源码version2.6.10)

1、new Vue的时候传了一个对象,把该对象记为options,Vue将options中自定义的属性和Vue构造函数中定义的属性合并为vm.$options,vm.$options.data()中的this指向vm.$options,而methodA和B并没有直接挂在vm.$options下,所以this.methodA和this.B为undefined。

// 创建一个vue实例
const options = {
  customOption: 'foo',
  data () {
    A: this.methodA
  },
  methods: {
    methodA () {}
  },
  created: function () {
    console.log(this.$options.customOption) // => 'foo'
  }
};
new Vue(options);
 
// src/core/instance/init.js
initMixin (Vue: Class<Component>) {
  const vm: Component = this
  // ...
  vm.$options = mergeOptions(
    resolveConstructorOptions(vm.constructor),
    options || {},
    vm
  )
  // ...
}

2、然后将vm.$options.data映射到vm._data,使得可以通过vm._data访问数据,在映射过程中,通过call将data()的this指向当前的实例vm,并将data()的执行结果返回,因为prop和methods的初始化在data之前,所以这时vm上已有_props和_methods,可以拿到this.methodA和this.B。(vm.key如何实现vm._props.key效果见3)。

// src/core/instance/state.js
initState (vm: Component) {
  // ...
  const opts = vm.$options
  if (opts.props) initProps(vm, opts.props)
  if (opts.methods) initMethods(vm, opts.methods)
  if (opts.data) {
    initData(vm) // 里面通过getData(data, vm)改变this
  }
  // ...
}
 
getData (data: Function, vm: Component): any {
  // ...
  try {
    return data.call(vm, vm) // this替换为vm
  }
  // ...
}

3、上面把属性映射到了vm._data里,可以通过vm._data.A访问数据,Vue再通过一个代理方法使得vm.A可以直接访问A。

// src/core/instance/state.js
proxy(vm, `_data`, key);
 
proxy (target: Object, sourceKey: string, key: string) {
  sharedPropertyDefinition.proxyget = function proxyGetter () {
    return this[sourceKey][key]
  }
  sharedPropertyDefinition.set = function proxySetter (val) {
    this[sourceKey][key] = val
  }
  Object.defineProperty(target, key, sharedPropertyDefinition)
}

总结

data()中若使用了this来访问props或methods,在重置$data时,注意this.$options.data()的this指向,最好使用this.$options.data.call(this)。

以上这篇Vue中的this.$options.data()和this.$data用法说明就是小编分享给大家的全部内容了,希望能给大家一个参考,也希望大家多多支持三水点靠木。

Javascript 相关文章推荐
番茄的表单验证类代码修改版
Jul 18 Javascript
JQuery 初体验(建议学习jquery)
Apr 25 Javascript
分享10篇优秀的jQuery幻灯片制作教程及应用案例
Apr 16 Javascript
jQuery验证Checkbox是否选中的代码 推荐
Sep 04 Javascript
JS添加删除一组文本框并对输入信息加以验证判断其正确性
Apr 11 Javascript
js实现改进的仿蓝色论坛导航菜单效果代码
Sep 06 Javascript
AngularJS中$apply方法和$watch方法用法总结
Dec 13 Javascript
JS实现给对象动态添加属性的方法
Jan 05 Javascript
jQuery实现CheckBox全选、全不选功能
Jan 11 Javascript
用JavaScript实现让浏览器停止载入页面的方法
Jan 19 Javascript
在vue项目中引用Iview的方法
Sep 14 Javascript
为什么要使用Vuex的介绍
Jan 19 Javascript
解决vue的router组件component在import时不能使用变量问题
Jul 26 #Javascript
Element Steps步骤条的使用方法
Jul 26 #Javascript
解决vue动态路由异步加载import组件,加载不到module的问题
Jul 26 #Javascript
vue 动态设置img的src地址无效,npm run build 后找不到文件的解决
Jul 26 #Javascript
VUE动态生成word的实现
Jul 26 #Javascript
Element Dialog对话框的使用示例
Jul 26 #Javascript
在vue中使用防抖函数组件操作
Jul 26 #Javascript
You might like
PHP实现简单数字分页效果
2015/07/26 PHP
PHP实现的memcache环形队列类实例
2015/07/28 PHP
php关键字仅替换一次的实现函数
2015/10/29 PHP
CI框架支持$_GET的两种实现方法
2016/05/18 PHP
PHP array_reverse() 函数原理及实例解析
2020/07/14 PHP
一份老外写的XMLHttpRequest代码多浏览器支持兼容性
2007/01/11 Javascript
用cookies实现的可记忆的样式切换效果代码下载
2007/12/24 Javascript
jQuery事件 delegate()使用方法介绍
2012/10/30 Javascript
基于jquery实现拆分姓名的方法(纯JS版)
2013/05/08 Javascript
JsRender for index循环索引用法详解
2014/10/31 Javascript
基于jQuery.Hz2Py.js插件实现的汉字转拼音特效
2015/05/07 Javascript
Jquery遍历select option和添加移除option的实现方法
2016/08/26 Javascript
创建简单的node服务器实例(分享)
2017/06/23 Javascript
vue 全选与反选的实现方法(无Bug 新手看过来)
2018/02/09 Javascript
Vue项目中跨域问题解决方案
2018/06/05 Javascript
jquery 验证用户名是否重复代码实例
2019/05/14 jQuery
layui自定义工具栏的方法
2019/09/19 Javascript
JS实现打砖块游戏
2020/02/14 Javascript
解决Vue watch里调用方法的坑
2020/11/07 Javascript
[02:02:38]VG vs Mineski Supermajor 败者组 BO3 第一场 6.6
2018/06/07 DOTA
[01:52]2020年DOTA2 TI10夏季活动预告片
2020/07/15 DOTA
Python学习笔记之os模块使用总结
2014/11/03 Python
Python数据结构与算法之使用队列解决小猫钓鱼问题
2017/12/14 Python
python打包压缩、读取指定目录下的指定类型文件
2018/04/12 Python
不管你的Python报什么错,用这个模块就能正常运行
2018/09/14 Python
Python正则表达式如何匹配中文
2020/05/27 Python
深圳茁壮笔试题
2015/05/28 面试题
大学军训感言600字
2014/02/25 职场文书
合作意向书范本
2014/03/31 职场文书
公司捐款倡议书
2014/05/14 职场文书
分公司经理任命书
2014/06/05 职场文书
客运企业隐患排查工作方案
2014/06/06 职场文书
爱护公共设施标语
2014/06/24 职场文书
2014大学生党员评议个人总结
2014/09/22 职场文书
水电工程师岗位职责
2015/02/13 职场文书
nginx日志格式分析和修改
2022/04/28 Servers