Vue3源码学习6-增强鲁棒性?


前言

“鲁棒性”(Robustness) 是一个软件工程术语,意思是:

程序在面对异常情况、边界条件或不完美输入时,仍能稳定、正确运行的能力。


✅ 举例说明

情况鲁棒性体现
传入已经是代理的对象不重复代理,直接返回
出现循环引用不死递归,正确引用自己
用户传入 null/undefined/非对象不报错,直接返回原值或做容错处理
属性不存在返回 undefined 而不是抛异常

✅ 在 Vue 响应式系统中的鲁棒性

你的 reactive() 方法如果没有处理好这些情况:

  • 传入已经代理过的对象又再次代理,会导致多层代理或性能浪费;
  • 遇到循环引用没做缓存,会死栈;
  • getset 时没做 null 检查,会报错。

这些都说明鲁棒性不够。


✅ 总结一句话:

鲁棒性 = 出错时不崩溃,边缘条件下仍合理运行。

一个高质量的框架/库,必须具备良好的鲁棒性。

下面在现有 reactive() 实现上,增加 4 项关键的鲁棒性增强,确保 mini-vue 面对各种异常/边界情况都能稳定运行。


✅ 1. 忽略非对象类型(null、number、boolean 等)

if (!isObject(target)) {
  return target
}

原因:

非对象是不能被 Proxy 代理的,否则直接报错。


✅ 2. 避免重复代理(支持 isReactive() 和 WeakMap 缓存双保险)

if ((target as any)[ReactiveFlags.IS_REACTIVE]) {
  return target // 已是响应式,直接返回
}

if (reactiveMap.has(target)) {
  return reactiveMap.get(target)
}

原因:

即使外部不小心再次调用 reactive(),也应返回已生成的代理对象。


✅ 3. 防止循环引用死递归:递归时也判断是否已经代理

if (isObject(result)) {
  return reactive(result)
}

需要进一步增强为:

if (isObject(result)) {
  // 优先使用 WeakMap 缓存中的代理
  const existingProxy = reactiveMap.get(result)
  return existingProxy ? existingProxy : reactive(result)
}

✅ 4. 兼容 toRaw() 还原原始对象

你已有:

if (key === ReactiveFlags.RAW) {
  return target
}

再配合:

export function toRaw(value: any): any {
  return value && (value as any)[ReactiveFlags.RAW] || value
}

这样你可以任意判断、恢复响应式对象。


🧩 最终鲁棒版 reactive.ts 核心结构如下:

export function reactive<T extends object>(target: T): T {
  if (!isObject(target)) {
    return target
  }

  if ((target as any)[ReactiveFlags.IS_REACTIVE]) {
    return target
  }

  if (reactiveMap.has(target)) {
    return reactiveMap.get(target)
  }

  const proxy = new Proxy(target, {
    get(target, key, receiver) {
      if (key === ReactiveFlags.IS_REACTIVE) return true
      if (key === ReactiveFlags.RAW) return target

      const result = Reflect.get(target, key, receiver)
      track(target, key)

      if (isObject(result)) {
        const existingProxy = reactiveMap.get(result)
        return existingProxy ? existingProxy : reactive(result)
      }

      return result
    },

    set(target, key, value, receiver) {
      const result = Reflect.set(target, key, value, receiver)
      trigger(target, key)
      return result
    }
  })

  reactiveMap.set(target, proxy)
  return proxy
}

✅ 附加:isObject 工具函数

function isObject(val: any): val is object {
  return val !== null && typeof val === 'object'
}

✅ 小结

防御策略解决的问题
非对象返回原值防止 Proxy 报错
重复代理检测避免多层代理 / 死递归
IS_REACTIVE 检查允许 isReactive() 判定状态
RAW 支持toRaw() 安全还原原始对象
WeakMap 缓存优化性能 + 支持循环引用

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

TE-茶叶蛋

踩坑不易,您的打赏,感谢万分

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值