当前位置: 首页 > news >正文

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

文章目录

  • 前言
    • ✅ 举例说明
    • ✅ 在 Vue 响应式系统中的鲁棒性
    • ✅ 总结一句话:
    • ✅ 1. **忽略非对象类型(null、number、boolean 等)**
      • 原因:
    • ✅ 2. **避免重复代理(支持 isReactive() 和 WeakMap 缓存双保险)**
      • 原因:
    • ✅ 3. **防止循环引用死递归:递归时也判断是否已经代理**
    • ✅ 4. **兼容 `toRaw()` 还原原始对象**
    • 🧩 最终鲁棒版 `reactive.ts` 核心结构如下:
    • ✅ 附加:`isObject` 工具函数
    • ✅ 小结


前言

“鲁棒性”(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 trueif (key === ReactiveFlags.RAW) return targetconst 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 缓存优化性能 + 支持循环引用

相关文章:

  • 网络安全:sql注入练习靶场——sqli_labs安装保姆级教程
  • ICML2021 | DeiT | 训练数据高效的图像 Transformer 与基于注意力的蒸馏
  • python中的函数
  • Python 装饰器优化策略模式:电商促销折扣的优雅解法
  • 【多线程】七、POSIX信号量 环形队列的生产者消费者模型
  • 防范反序列化攻击:如何安全处理敏感数据序列化
  • 【ICMP协议深度解析】从网络诊断到安全实践
  • 华为OD机试真题——智能驾驶(2025A卷:200分)Java/python/JavaScript/C/C++/GO最佳实现
  • 嵌入式Linux应用项目----智能网关
  • 【tcp连接windows redis】
  • 威科夫操盘手法笔记
  • 游戏代码C
  • 使用DeepSeek定制Python小游戏——以“俄罗斯方块”为例
  • Leetcode刷题记录28——缺失的第一个正数
  • 优化高搜索量还是低竞争关键词?SEO策略解析
  • 神经网络的基本概念与深度解析——基于生物机制的仿生建模与工程实现
  • Redis 脚本:深入理解与实践指南
  • LabVIEW三轴电机控制
  • Python函数参数机制深度解析与最佳实践
  • 用docker ffmpeg测试视频vmaf分数,很快不用编译
  • 降准又降息!央行发布3类10项措施
  • 国际上首次,地月空间卫星激光测距试验在白天成功实施
  • 以色列计划“占领加沙”,特朗普下周中东行结束之际将是“机会窗口”
  • 习近平将对俄罗斯进行国事访问并出席纪念苏联伟大卫国战争胜利80周年庆典
  • “矿茅”国际化才刚开始?紫金矿业拟分拆境外黄金矿山资产于港交所上市
  • 体坛联播|曼联一只脚迈进欧联杯决赛,赵心童4比4奥沙利文