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 相关文章推荐
实例:尽可能写友好的Javascript代码
Oct 09 Javascript
javascript 当前日期转化为中文的实现代码
May 13 Javascript
javascript开发技术大全-第3章 js数据类型
Jul 03 Javascript
ASP.NET jQuery 实例10 动态修改hyperlink的URL值
Feb 03 Javascript
Dom 学习总结以及实例的使用介绍
Apr 24 Javascript
javascript实现存储hmtl字符串示例
Apr 25 Javascript
javascript移动设备Web开发中对touch事件的封装实例
Jun 05 Javascript
jquery+css实现动感的图片切换效果
Nov 25 Javascript
详解Javascript中new()到底做了些什么?
Mar 29 Javascript
vue 不使用select实现下拉框功能(推荐)
May 17 Javascript
JS计算斐波拉切代码实例
Sep 12 Javascript
vue实现简单的登录弹出框
Oct 26 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与flash as3 socket通信传送文件实现代码
2014/08/16 PHP
js+php实现静态页面实时调用用户登陆状态的方法
2015/01/04 PHP
浅谈PHP中foreach/in_array的使用
2015/11/02 PHP
yii权限控制的方法(三种方法)
2015/12/28 PHP
PHP生成各种常见验证码和Ajax验证过程
2016/01/10 PHP
phpstorm最新激活码分享亲测phpstorm2020.2.3版可用
2020/11/22 PHP
彪哥1.1(智能表格)提供下载
2006/09/07 Javascript
jquery 插件 任意位置浮动固定层
2008/12/25 Javascript
JQuery里面的几种选择器 查找满足条件的元素$(&quot;#控件ID&quot;)
2011/08/23 Javascript
jQuery学习笔记之jQuery中的$
2015/01/19 Javascript
javascript中mouseover、mouseout使用详解
2015/07/19 Javascript
AngularJS ng-bind-html 指令详解及实例代码
2016/07/30 Javascript
使用jquery.qrcode.js生成二维码插件
2016/10/17 Javascript
AngularJS自定义控件实例详解
2016/12/13 Javascript
JS/jquery实现一个网页内同时调用多个倒计时的方法
2017/04/27 jQuery
轻松玩转BootstrapTable(后端使用SpringMVC+Hibernate)
2017/09/06 Javascript
vue中v-cloak解决刷新或者加载出现闪烁问题(显示变量)
2018/04/20 Javascript
js实现二级菜单点击显示当前内容效果
2018/04/28 Javascript
Vue实现拖放排序功能的实例代码
2019/07/08 Javascript
layui清除radio的选中状态实例
2019/11/14 Javascript
vue微信分享插件使用方法详解
2020/02/18 Javascript
[01:54]胎教DOTA2 准妈妈玩家现身中国区预选赛
2016/06/26 DOTA
Python 返回汉字的汉语拼音
2009/02/27 Python
python通过ssh-powershell监控windows的方法
2015/06/02 Python
python pandas中DataFrame类型数据操作函数的方法
2018/04/08 Python
Python二次规划和线性规划使用实例
2019/12/09 Python
对tensorflow中cifar-10文档的Read操作详解
2020/02/10 Python
你需要学会的8个Python列表技巧
2020/06/24 Python
详解HTML5 canvas绘图基本使用方法
2018/01/29 HTML / CSS
html5 video全屏播放/自动播放的实现示例
2020/08/06 HTML / CSS
枚举与#define宏的区别
2014/04/30 面试题
网络研修随笔感言
2014/02/17 职场文书
2014年社区国庆节活动方案
2014/09/16 职场文书
2014年外贸业务员工作总结
2014/12/11 职场文书
高校自主招生教师推荐信
2015/03/23 职场文书
新手必备Python开发环境搭建教程
2021/05/28 Python