Vue配置项data

data

目录

  • data
    • 目录
    • 类型介绍
    • 关键原理
    • 编译过程
      • Vue2
      • Vue3

📌Vue.js 中的 data(Obj/Function)属性是 Vue 实例的一个配置选项

类型介绍

  • 对象式

    对于根实例或者非复用组件,通常直接提供一个对象字面量作为 data 的值。在对象式中,data属性是一个普通的对象,并且直接作为组件实例的一个属性

    // 对象形式 (适用于根实例或非复用组件)
    new Vue({data: {message: 'Hello, Vue!',count: 0,user: {name: 'John Doe',age: 30}},// 其他选项...
    })
    • 区别:data 是一个普通的对象,通过对象属性初始化数据
    • 优点:
      • 语法简洁,直观易懂。可以通过对象字面量的方式一次性定义多个属性,方便管理多个数据。
    • 缺点:
      • 数据共享风险:所有组件实例共享同一个数据对象,如果 data 仍然是一个纯粹的对象,当多个组件修改同一个属性时,则所有的实例将共享引用同一个数据对象!可能会造成意外的数据污染,不方便追踪数据的变化来源。
  • 函数式中,data属性是一个返回一个对象的函数。对于复用型组件(如.vue文件),推荐使用函数形式来返回一个新对象实例

    // 函数形式 (适用于复用组件)
    Vue.component('my-component', {data() {return {localMessage: 'Hello from Component!'
    //此处的this是Vue实例对象,data函数不能写成箭头函数,不然this会指向window};}},// 其他选项...
    })
    • 区别:data 是一个返回包含数据对象的函数
    • 优点:
      • 数据独立性:每个组件实例都有独立的数据对象,避免了数据共享风险
      • 可以在函数中进行复杂的数据处理或计算,对数据进行响应式处理
    • 缺点:
      • 代码略显繁琐,**需要使用 **return 关键字返回数据对象
      • 不如对象式写法直观,特别是当需要定义多个属性时,需要书写更多的代码
  • 限制: 组件的定义只接受 function

  • 访问与更新

    在模板中,可以通过 Mustache 语法 ({{ }}) 来显示 data 中的数据。同时,Vue 提供了响应式系统,任何通过Vue实例方法或计算属性对 data 的更改都会触发相应的视图更新

  • 注意事项

    • Vue管理的函数,一定不要写箭头函数,一旦写了箭头函数,this就不再是Vue实例了。用来定义要在当前vue实例中使用的数据

关键原理

  1. Vue实例的数据对象。Vue将会递归将dataproperty 转换为 getter/setter,从而让 dataproperty能够响应数据变化(后续添加的不会有变化,除非使用Vue.$set)。浏览器 API创建的原生对象,原型上的 property会被忽略。vue对于数组也有特殊的变更检测机制(仅仅是修改数组的几个方法,vue进行了重写)
  2. 实例创建之后,可以通过 vm.$data 访问原始数据对象。Vue 实例代理了 data 对象上的所有属性,因此通过 vm.a 访问等同于 vm.$data.a
  3. _$ 开头的 property不会Vue 实例代理,因为它们可能和 Vue 内置的 propertyAPI 方法冲突。你可以使用例如 vm.$data._property 的方式访问这些 property
  4. 通过提供 data 函数,每次创建一个新实例后,能够调用 data 函数,以便返回一个全新副本的初始数据对象。如果需要,可以通过将 vm.$data 传入 JSON.parse(JSON.stringify(...)) 来获得原始数据对象的深拷贝

编译过程

📌Vue.js 中的 dataVue 实例的核心配置选项之一,它用于定义组件或实例的状态(数据)。在Vue中,数据与视图是双向绑定的,这意味着当 data 中的数据发生变化时,依赖这些数据的DOM元素会自动更新;反之,用户对视图中的表单元素进行交互,也会相应地更新 data 中的数据

Vue2

1. Vue 实例初始化

在创建 Vue 实例时,会调用 _init 方法,该方法会初始化各种选项,包括 data

Vue.prototype._init = function (options) {const vm = this;vm.$options = mergeOptions(resolveConstructorOptions(vm.constructor), options || {});initState(vm);// ...
};

2. 初始化状态

initState 方法会根据组件选项初始化 propsmethodsdatacomputedwatch

function initState(vm) {const opts = vm.$options;if (opts.data) {initData(vm);}// ...
}

3. 初始化数据

initData 方法负责初始化 data 选项并将其转换为响应式数据。

function initData(vm) {let data = vm.$options.data;data = vm._data = typeof data === 'function' ? data.call(vm, vm) : data || {};for (const key in data) {proxy(vm, `_data`, key);}observe(data, true /* asRootData */);
}

4. 数据代理

proxy 方法将 data 对象上的属性代理到 Vue 实例上,使得可以通过 this.property 直接访问 data 中的数据。

function proxy(target, sourceKey, key) {Object.defineProperty(target, key, {configurable: true,enumerable: true,get() {return target[sourceKey][key];},set(val) {target[sourceKey][key] = val;}});
}

5. 数据观察(响应式)

observe 方法将数据对象转换为响应式对象,使用 Observer 类来实现。

function observe(value, asRootData) {if (!isObject(value) || value instanceof VNode) {return;}let ob;if (hasOwn(value, '__ob__') && value.__ob__ instanceof Observer) {ob = value.__ob__;} else if (shouldObserve &&(Array.isArray(value) || isPlainObject(value)) &&Object.isExtensible(value) &&!value._isVue) {ob = new Observer(value);}if (asRootData && ob) {ob.vmCount++;}return ob;
}

6. 定义响应式属性

defineReactive 方法将对象的每个属性转换为响应式,通过 Object.defineProperty 实现。

function defineReactive(obj, key, val, customSetter, shallow) {const dep = new Dep();const property = Object.getOwnPropertyDescriptor(obj, key);if (property && property.configurable === false) {return;}const getter = property && property.get;const setter = property && property.set;if ((!getter || setter) && arguments.length === 2) {val = obj[key];}let childOb = !shallow && observe(val);Object.defineProperty(obj, key, {enumerable: true,configurable: true,get() {const value = getter ? getter.call(obj) : val;if (Dep.target) {dep.depend();if (childOb) {childOb.dep.depend();if (Array.isArray(value)) {dependArray(value);}}}return value;},set(newVal) {const value = getter ? getter.call(obj) : val;if (newVal === value || (newVal !== newVal && value !== value)) {return;}if (setter) {setter.call(obj, newVal);} else {val = newVal;}childOb = !shallow && observe(newVal);dep.notify();}});
}

7. 依赖收集和变化通知

Dep 类用于依赖收集和变化通知。

class Dep {constructor() {this.id = uid++;this.subs = [];}addSub(sub) {this.subs.push(sub);}removeSub(sub) {remove(this.subs, sub);}depend() {if (Dep.target) {Dep.target.addDep(this);}}notify() {const subs = this.subs.slice();for (let i = 0, l = subs.length; i < l; i++) {subs[i].update();}}
}Dep.target = null;

8. Watcher

Watcher 类用于更新视图,当数据变化时调用更新函数。

class Watcher {constructor(vm, expOrFn, cb, options) {this.vm = vm;vm._watchers.push(this);this.getter = expOrFn;this.cb = cb;this.value = this.get();}get() {Dep.target = this;const value = this.getter.call(this.vm, this.vm);Dep.target = null;return value;}addDep(dep) {dep.addSub(this);}update() {const value = this.get();if (value !== this.value) {const oldValue = this.value;this.value = value;this.cb.call(this.vm, value, oldValue);}}
}

以下是一个简单的实现示例:

class Vue {constructor(options) {this.$options = options;this._data = options.data();this.init();}init() {initState(this);}
}function initState(vm) {let data = vm._data;for (let key in data) {proxy(vm, '_data', key);}observe(data);
}function proxy(target, sourceKey, key) {Object.defineProperty(target, key, {enumerable: true,configurable: true,get() {return target[sourceKey][key];},set(val) {target[sourceKey][key] = val;}});
}function observe(value) {if (!isObject(value)) return;new Observer(value);
}function isObject(obj) {return obj !== null && typeof obj === 'object';
}class Observer {constructor(value) {this.value = value;if (Array.isArray(value)) {} else {this.walk(value);}}walk(obj) {for (let key in obj) {defineReactive(obj, key, obj[key]);}}
}function defineReactive(obj, key, val) {const dep = new Dep();let childOb = observe(val);Object.defineProperty(obj, key, {enumerable: true,configurable: true,get() {if (Dep.target) {dep.depend();}return val;},set(newVal) {if (newVal === val) return;val = newVal;childOb = observe(newVal);dep.notify();}});
}class Dep {constructor() {this.subs = [];}addSub(sub) {this.subs.push(sub);}depend() {if (Dep.target) {Dep.target.addDep(this);}}notify() {const subs = this.subs.slice();for (let i = 0; i < subs.length; i++) {subs[i].update();}}
}Dep.target = null;class Watcher {constructor(vm, expOrFn, cb) {this.vm = vm;this.getter = expOrFn;this.cb = cb;this.value = this.get();}get() {Dep.target = this;const value = this.getter.call(this.vm, this.vm);Dep.target = null;return value;}addDep(dep) {dep.addSub(this);}update() {const value = this.get();if (value !== this.value) {const oldValue = this.value;this.value = value;this.cb.call(this.vm, value, oldValue);}}
}const vm = new Vue({data() {return {message: 'Hello Vue!'};}
});new Watcher(vm, function () {console.log(`Message is: ${this.message}`);
}, (newVal, oldVal) => {console.log(`Updated message from ${oldVal} to ${newVal}`);
});vm.message = 'Hello World!';

这个示例展示了 Vue 2 中 data 选项的底层实现逻辑,包括响应式系统的核心原理。

Vue3

Vue 的 data 选项底层逻辑主要涉及响应式系统的实现,它包括数据初始化、数据代理、依赖收集和变化通知等。以下是详细的实现过程:

1. 数据初始化

在 Vue 实例初始化时,会调用 _init 方法,其中会初始化组件的 data 选项。

Vue.prototype._init = function (options) {const vm = this;// 合并选项vm.$options = mergeOptions(resolveConstructorOptions(vm.constructor), options || {});// 初始化数据initState(vm);// ...
};

2. 初始化状态

initState 方法会根据组件选项初始化 propsmethodsdatacomputedwatch

function initState(vm) {const opts = vm.$options;if (opts.data) {initData(vm);}// ...
}

3. 初始化数据

initData 方法负责初始化 data 选项并将其转换为响应式数据。

function initData(vm) {let data = vm.$options.data;// 支持函数形式的 datadata = vm._data = typeof data === 'function' ? data.call(vm, vm) : data || {};// 将 data 代理到 vm 上for (const key in data) {proxy(vm, `_data`, key);}// 观察数据,转换为响应式observe(data, true);
}

4. 数据代理

proxy 方法将 data 对象上的属性代理到 Vue 实例上,使得可以通过 this.property 直接访问 data 中的数据。

function proxy(target, sourceKey, key) {Object.defineProperty(target, key, {configurable: true,enumerable: true,get() {return target[sourceKey][key];},set(val) {target[sourceKey][key] = val;}});
}

5. 数据观察(响应式)

observe 方法将数据对象转换为响应式对象,使用 Observer 类来实现。

function observe(value, asRootData) {if (!isObject(value)) return;let ob;if (hasOwn(value, '__ob__') && value.__ob__ instanceof Observer) {ob = value.__ob__;} else {ob = new Observer(value);}return ob;
}class Observer {constructor(value) {this.value = value;this.dep = new Dep();def(value, '__ob__', this);if (Array.isArray(value)) {// 处理数组protoAugment(value, arrayMethods);this.observeArray(value);} else {this.walk(value);}}walk(obj) {for (const key in obj) {defineReactive(obj, key);}}observeArray(items) {for (let i = 0, l = items.length; i < l; i++) {observe(items[i]);}}
}

6. 定义响应式属性

defineReactive 方法将对象的每个属性转换为响应式,通过 Object.defineProperty 实现。

function defineReactive(obj, key, val) {const dep = new Dep();const property = Object.getOwnPropertyDescriptor(obj, key);if (property && property.configurable === false) return;const getter = property && property.get;const setter = property && property.set;if ((!getter || setter) && arguments.length === 2) {val = obj[key];}let childOb = observe(val);Object.defineProperty(obj, key, {enumerable: true,configurable: true,get() {const value = getter ? getter.call(obj) : val;if (Dep.target) {dep.depend();if (childOb) {childOb.dep.depend();if (Array.isArray(value)) {dependArray(value);}}}return value;},set(newVal) {const value = getter ? getter.call(obj) : val;if (newVal === value || (newVal !== newVal && value !== value)) return;if (setter) {setter.call(obj, newVal);} else {val = newVal;}childOb = observe(newVal);dep.notify();}});
}

7. 依赖收集和变化通知

Dep 类用于依赖收集和变化通知。

class Dep {constructor() {this.subs = [];}addSub(sub) {this.subs.push(sub);}removeSub(sub) {remove(this.subs, sub);}depend() {if (Dep.target) {Dep.target.addDep(this);}}notify() {const subs = this.subs.slice();for (let i = 0, l = subs.length; i < l; i++) {subs[i].update();}}
}Dep.target = null;

8. Watcher

Watcher 类用于更新视图,当数据变化时调用更新函数。

class Watcher {constructor(vm, expOrFn, cb, options) {this.vm = vm;vm._watchers.push(this);this.getter = expOrFn;this.cb = cb;this.value = this.get();}get() {Dep.target = this;const value = this.getter.call(this.vm, this.vm);Dep.target = null;return value;}addDep(dep) {dep.addSub(this);}update() {const value = this.get();if (value !== this.value) {const oldValue = this.value;this.value = value;this.cb.call(this.vm, value, oldValue);}}
}

以下是一个简单的实现示例:

// 模拟 Vue
class Vue {constructor(options) {this.$options = options;this._data = options.data();this.init();}init() {initState(this);}
}function initState(vm) {let data = vm._data;// 代理for (let key in data) {proxy(vm, '_data', key);}// 观察数据observe(data);
}function proxy(target, sourceKey, key) {Object.defineProperty(target, key, {enumerable: true,configurable: true,get() {return target[sourceKey][key];},set(val) {target[sourceKey][key] = val;}});
}function observe(value) {if (!isObject(value)) return;new Observer(value);
}function isObject(obj) {return obj !== null && typeof obj === 'object';
}class Observer {constructor(value) {this.value = value;if (Array.isArray(value)) {// 处理数组} else {this.walk(value);}}walk(obj) {for (let key in obj) {defineReactive(obj, key, obj[key]);}}
}function defineReactive(obj, key, val) {const dep = new Dep();let childOb = observe(val);Object.defineProperty(obj, key, {enumerable: true,configurable: true,get() {if (Dep.target) {dep.depend();}return val;},set(newVal) {if (newVal === val) return;val = newVal;childOb = observe(newVal);dep.notify();}});
}class Dep {constructor() {this.subs = [];}addSub(sub) {this.subs.push(sub);}depend() {if (Dep.target) {Dep.target.addDep(this);}}notify() {const subs = this.subs.slice();for (let i = 0; i < subs.length; i++) {subs[i].update();}}
}Dep.target = null;class Watcher {constructor(vm, expOrFn, cb) {this.vm = vm;this.getter = expOrFn;this.cb = cb;this.value = this.get();}get() {Dep.target = this;const value = this.getter.call(this.vm, this.vm);Dep.target = null;return value;}addDep(dep) {dep.addSub(this);}update() {const value = this.get();if (value !== this.value) {const oldValue = this.value;this.value = value;this.cb.call(this.vm, value, oldValue);}}
}// 使用示例
const vm = new Vue({data() {return {message: 'Hello Vue!'};}
});new Watcher(vm, function () {console.log(`Message is: ${this.message}`);
}, (newVal, oldVal) => {console.log(`Updated message from ${oldVal} to ${newVal}`);
});vm.message = 'Hello World!';

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/bicheng/35771.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Follow Carl To Grow|【LeetCode】491.递增子序列,46.全排列,47.全排列 II

【LeetCode】491.递增子序列 题意&#xff1a;给你一个整数数组 nums &#xff0c;找出并返回所有该数组中不同的递增子序列&#xff0c;递增子序列中 至少有两个元素 。你可以按 任意顺序 返回答案。 数组中可能含有重复元素&#xff0c;如出现两个整数相等&#xff0c;也可以…

dataguard 主备切换方式switchover 和 failover 操作步骤

作者介绍&#xff1a;老苏&#xff0c;10余年DBA工作运维经验&#xff0c;擅长Oracle、MySQL、PG数据库运维&#xff08;如安装迁移&#xff0c;性能优化、故障应急处理等&#xff09; 公众号&#xff1a;老苏畅谈运维 欢迎关注本人公众号&#xff0c;更多精彩与您分享。datagu…

SpringBoot整合拦截器和日期转换器

一、SpringBoot整合拦截器 1.添加拦截器 package com.by.interceptor;import com.by.pojo.User; import org.springframework.stereotype.Component; import org.springframework.web.servlet.HandlerInterceptor;import javax.servlet.http.HttpServletRequest; import java…

一文读懂充电桩平台如何实现分账:汇付是如何实现分账的

汇付实现分账的方式是通过多方分账功能和延时分账模式&#xff0c;结合接口与控台操作来实现灵活的资金分配和高效结算。 【慧哥开源充电桩平台】下载源码地址 https://liwenhui.blog.csdn.net/article/details/134773779?spm1001.2014.3001.5502 在当前支付环境中&#xff0c…

【ARM】内存属性Memory Attributes (MemAttr)

目录 1. EWA 2. Device 3. Cacheable 4. Allocate 5. 内存属性的传播 6. 事务属性组合 7. Memory Type 内存属性Memory Attributes (MemAttr) 包含Early Write Acknowledgment (EWA), Device, Cacheable, 以及Allocate。 1. EWA EWA&#xff0c;Early Write Acknowledg…

AI自动生成角色和情节连续的漫画,中山大学联想提出AutoStudio,可以多轮交互式连续生成并保持主题一致性。

中山大学和联想研究院提出AutoStudio: 是一种无需训练的多代理框架&#xff0c;用于多轮交互式图像生成&#xff0c;能够在生成多样化图像的同时保持主体一致性。 AutoStudio 采用三个基于 LLM 的智能体来解释人类意图并为 SD 模型生成适当的布局指导。此外&#xff0c;还引入…

微软代码页标识符 (Code Page Identifiers)

代码页标识符 (Code Page Identifiers) 双语对照 Identifiere标识符.NET Name.NET 名称Additional information其他信息037IBM037IBM EBCDIC US-CanadaIBM EBCDIC US-Canada437IBM437OEM United StatesOEM 美国500IBM500IBM EBCDIC InternationalIBM EBCDIC 国际字符集708ASMO…

让AI保持怪异

让AI保持怪异 Anthropic的创意技术专家和员工设计师凯尔图尔曼(Kyle Turman)分享了一种深深引起共鸣的观点。他说(转述原话):“人工智能实际上真的很奇怪&#xff0c;我认为人们对这一点的认识还不够。”这引发了我向小组提出的问题:我们是否有消毒人工智能固有的陌生感的风险?…

python-序列相关

序列&#xff08;squence&#xff09;是一组按顺序、紧密排列在一起的数据集。序列的作用是便于管理、方便数据操作更重要的是序列支持切片操作。 序列主要包括&#xff1a;列表、元组、字符串和字节串 内置数据结构&#xff1a; 容器&#xff1a;列表、元组、字典、集合 结构…

Chromium 调试指南2024 Mac篇 - 准备工作 (一)

1.引言 Chromium是一个由Google主导开发的开源浏览器项目&#xff0c;它为Google Chrome浏览器提供了基础框架。Chromium不仅是研究和开发现代浏览器技术的重要平台&#xff0c;还为众多其他基于Chromium的浏览器&#xff08;如Microsoft Edge、Brave等&#xff09;提供了基础…

atcoder abc 359

A count takahashi 问题: 思路&#xff1a;字符串比较 代码&#xff1a; #include <bits/stdc.h>using namespace std;int main() {int n;cin >> n;int ans 0;for(int i 1; i < n; i ) {string s;cin >> s;if(s[0] T) ans ;}cout << ans;re…

命令ifconfig-显示网络接口的当前状态

ifconfig是一个在类Unix系统中用于配置网络接口的命令行工具。它可以用来显示网络接口的当前状态&#xff0c;或者设置网络接口的参数&#xff0c;如IP地址、子网掩码、MAC地址等。 基本用法 显示所有网络接口的信息&#xff1a; ifconfig显示特定网络接口的信息&#xff1a…

Android 多媒体开发——Media3与MediaSession最全使用指南

一、Media3库简介 1.1 Media3是什么&#xff1f; 官方释义&#xff1a; Jetpack Media3 is the new home for media libraries that enables Android apps to display rich audio and visual experiences. Media3 offers a simple architecture with powerful customization,…

软考出成绩了,速查,查分方式看这里

各位考生&#xff0c;软考出成绩啦&#xff01;大家赶紧查一下&#xff0c;各科都45分就是通过&#xff01; 软考成绩查询入口已开通&#xff0c;此刻你是不是既激动又忐忑&#xff1f;速查&#xff01; ★ 查询网站 中国计算机技术职业资格网https://bm.ruankao.org.cn/sign/…

如何轻松获取 GitLab 指定分支特定路径下的文件夹内容

第一步&#xff1a; 获取 accessToken 及你的 项目 id &#xff1a; 获取 accessToken ,点击用户头像进入setting 按图示操作&#xff0c;第 3 步 填写你发起请求的域名。 获取项目 id , 简单粗暴方案 进入 你项目仓库页面后 直接 源码搜索 project_id&#xff0c; value 就…

记录Gstreamer的uridecodebin可以自动选择硬解码器

记录&#xff1a; uridecodebin3 和uridecodebin优先硬解码 这两个插件&#xff0c;本来是负责动态选择合适的解码器来处理特定的媒体流&#xff0c;使用案例&#xff1a; gst-launch-1.0 uridecodebin urirtsp://192.168.1.120:8554/test ! glimagesink -v gst-launch-1.0 …

Linux通用LInux高危漏洞(CVE-2024-1086)修复案例

一、漏洞描述 2024年3月28日&#xff0c;监 Linux kernel权限提升漏洞&#xff08;CVE-2024-1086&#xff09;的PoC/EXP在互联网上公开&#xff0c;该漏洞的CVSS评分为7.8&#xff0c;目前漏洞细节已经公开披露&#xff0c;美国网络安全与基础设施安全局&#xff08;CISA&…

Java高手的30k之路|面试宝典|精通项目介绍方法优化简历项目介绍

常用项目介绍结构及原因 以下是几个常用的项目介绍结构&#xff0c;以及为什么使用这些结构的原因&#xff1a; 1. STAR 方法 Situation&#xff08;情境&#xff09;&#xff1a;项目的背景和情况。 Task&#xff08;任务&#xff09;&#xff1a;你需要完成的任务。 Actio…

每天一学(2)

目录 1、线程池是如何知道线程任务是否完成 2、阻塞队列的有界和无界 3、ConcurrentHashMap底层实现原理 5、CAS机制 6、wait 和 notify 为什么要放在 synchronized 1、线程池是如何知道线程任务是否完成 线程池内部&#xff1a; 当把任务丢给线程池去执行&#xff0c;调度工作…

【UE5.3】笔记4-自定义材质蓝图

正常来说&#xff0c;我们都是拿到什么材质用什么材质&#xff0c;那么我们如何去创建自定义的材质呢&#xff1f; 首先&#xff0c;创建MyMaterials文件夹用来存放我们自制的材质&#xff1b; 然后&#xff0c;右键创建一个材质&#xff0c;起个名字&#xff0c;双击打开&am…