vue中$set原理

Vue 中的 $set 方法(Vue.set)主要用于 向响应式对象中添加一个新的属性,并确保这个新属性是响应式的,能够触发视图更新。


📌 背景问题:为什么需要 $set

在 Vue 2 中,直接给对象新增属性不会触发视图更新,例如:

复制代码
this.obj.a = 123 // 如果 a 原本不存在,则不会触发视图更新

这是因为 Vue 2 是基于 Object.defineProperty 来实现响应式系统的,它在对象初始化时只能"劫持"已有属性,无法监控新增的属性


$set 的作用

复制代码
this.$set(obj, key, value)

或者

复制代码
Vue.set(obj, key, value)

它的作用是:

  1. 为对象添加新属性

  2. 让这个新属性是响应式的

  3. 触发视图更新


🧠 原理解析(Vue 2)

复制代码
Vue.set = function (target, key, val) {
  if (Array.isArray(target) && isValidIndex(key)) {
    // 对数组使用 splice 替代设置
    target.splice(key, 1, val)
    return val
  }

  if (key in target && !(key in Object.prototype)) {
    target[key] = val
    return val
  }

  const ob = target.__ob__  // 这是 Observer 实例
  if (!ob) {
    target[key] = val
    return val
  }

  defineReactive(ob.value, key, val) // 核心:为新属性定义响应式
  ob.dep.notify()  // 通知依赖更新
  return val
}

核心步骤:

  1. 判断是否是数组 :数组用 splice 替代赋值(因为 Vue 对数组的监听基于方法拦截)。

  2. 通过 defineReactive() 为新属性添加 getter/setter,注入依赖收集逻辑。

  3. 通知依赖更新视图(dep.notify()


🧪 示例

复制代码
<template>
  <div>{{ user.name }}</div>
  <button @click="addProp">添加新属性</button>
</template>

<script>
export default {
  data() {
    return {
      user: {}
    }
  },
  methods: {
    addProp() {
      this.$set(this.user, 'name', '张三') // 触发视图更新
    }
  }
}
</script>

🆕 Vue 3 呢?

Vue 3 中,由于使用了 Proxy 实现响应式,不需要 $set,你可以直接赋值:

复制代码
reactiveObj.newKey = 123 // ✅ 直接响应式,无需 $set

🧩 总结

项目 Vue 2 Vue 3
添加响应式属性 需要 $set 不需要,直接赋值即可
实现机制 Object.defineProperty + defineReactive Proxy 拦截所有操作
数组监听 依赖重写 push/splice 等方法 Proxy 完整拦截

需要我帮你用简单代码实现一个简化版的 $set 吗?可以更好理解它的内部逻辑。

相关推荐
anyup1 小时前
2026第一站:分享我在高德大赛现场学到的技术、产品与心得
前端·架构·harmonyos
BBBBBAAAAAi1 小时前
Claude Code安装记录
开发语言·前端·javascript
xiaolyuh1232 小时前
【XXL-JOB】 GLUE模式 底层实现原理
java·开发语言·前端·python·xxl-job
源码获取_wx:Fegn08952 小时前
基于 vue智慧养老院系统
开发语言·前端·javascript·vue.js·spring boot·后端·课程设计
毕设十刻2 小时前
基于Vue的人事管理系统67zzz(程序 + 源码 + 数据库 + 调试部署 + 开发环境配置),配套论文文档字数达万字以上,文末可获取,系统界面展示置于文末
前端·数据库·vue.js
anyup2 小时前
从赛场到产品:分享我在高德大赛现场学到的技术、产品与心得
前端·harmonyos·产品
QQ19632884752 小时前
ssm基于Springboot+的球鞋销售商城网站vue
vue.js·spring boot·后端
前端工作日常2 小时前
我学习到的A2UI的功能:纯粹的UI生成
前端
Jing_Rainbow2 小时前
【 前端三剑客-37 /Lesson61(2025-12-09)】JavaScript 内存机制与执行原理详解🧠
前端·javascript·程序员
UIUV3 小时前
模块化CSS学习笔记:从作用域问题到实战解决方案
前端·javascript·react.js