1.Vue3源码系列 (一) watch
2.Vue3源码系列 (四) ref
3.源码视角,式源Vue3为什么推荐使用ref而不是码响reactive
Vue3源码系列 (一) watch
本文深入解析 Vue3 中 watch 的机制。首先,应式源码我们了解 watch 接收三个参数:监听的式源数据源 source、回调 cb 以及可选的码响 options。options 包括 immediate、应式源码mxnet 源码编译deep、式源flush、码响onTrack 和 onTrigger,应式源码用于控制立即执行、式源深度监听、码响回调时机以及收集依赖和触发更新时的应式源码自定义函数。回调 cb 接收 value、式源oldValue 和 onCleanUp 参数,码响用于执行特定操作,应式源码如响应表格页码变化重新请求数据,并在副作用清理时调用 onCleanUp 函数。
watch 支持监听单个数据或多个数据,其参数类型包括 WatchSource、响应式对象、源码之家表单MultiWatchSources 和 Readonly。单个数据源可以是 WatchSource 或响应式的对象,多个数据源则为 MultiWatchSources 或 Readonly。
watch 的核心在于 doWatch 函数,它接收与 watch 类似的参数。在源码中,doWatch 负责实现 watch 的逻辑。首先,它会检查是否提供了回调函数 cb。如果没有,且 options 中设置了 immediate 和 deep,会抛出警告,因为这些选项只对有回调的 doWatch 签名有效。接着,设置 getter,并配置强制触发和深度监听。根据 source 的类型,doWatch 进行不同的处理。
在处理源数据后,购买网址源码doWatch 会创建 effect,这是 Vue3 中实现响应式的关键。effect 通过 getter 获取当前值,然后在回调函数中使用 newValue 和 oldValue。这使得 watch 能在数据变化时触发回调函数,执行相应的操作。
总结,本文详细阐述了 Vue3 中 watch 的工作原理,从参数类型、回调函数到核心实现 doWatch 函数,全面深入地解析了 watch 的机制,帮助开发者更好地理解和运用 Vue3 的响应式特性。通过本文,读者可以深入了解 Vue3 watch 的内部工作流程,为构建高效、响应式的 Vue 应用提供技术支持。
Vue3源码系列 (四) ref
一般而言,reactive用于定义响应式对象,而ref则用于定义响应式原始值。奇迹源码合成前文已介绍reactive,了解到通过Proxy对目标对象进行代理实现响应式,非对象原始值的响应式问题则由ref解决。
ref和shallowRef各有三种重载,参数不同,都返回Ref/ShallowRef类型的值。createRef函数用于创建响应式值,类似reactive,createRef也是通过createReactiveObject创建响应式对象。而createRef返回RefImpl实例。
RefImpl是ref的核心内容,构造函数接收两个参数,value是传入的原始值,__v_isShallow用于区分深层/浅层响应式,isShallow()函数利用这个属性做判断。在Ref中,_value属性存储实际值,dep属性存储依赖,在class的swoole源码安装getter中通过trackRefValue(this)收集依赖,在setter中调用triggerRefValue(this, newVal)。
trackRefValue用于收集Ref依赖,接收RefBase类型值,在ref函数中接收RefImpl实例。shouldTrack用于暂停和恢复捕获依赖的标志,activeEffect标记当前活跃的effect。内部调用trackEffects函数收集依赖,该函数来自effect模块。
triggerRefValue函数用于触发Ref的响应式更新,triggerEffects函数来自effect模块。
Vue3还提供了自定义的Ref,可以传入getter和setter,自由选择track和trigger时机。
在setup函数中返回参数时,使用toRef创建ObjectRefImpl实例对响应式对象的某个属性进行解构。
ObjectRefImpl通过_object属性引用原始响应式对象,在getter中通过_object访问值,依赖收集由_object完成;在setter中,通过引用_object达到赋值操作,从而在_object中触发更新。toRef判断入参是否是Ref,是则直接返回,否则返回ObjectRefImpl。toRefs对传入的对象/数组进行遍历并执行toRef解构。
源码视角,Vue3为什么推荐使用ref而不是reactive
ref和reactive是Vue3中实现响应式数据的核心API。ref用于封装基本数据类型,而reactive用于处理对象和数组。尽管reactive似乎更适合处理对象,但Vue3官方文档更推荐使用ref。
官方文档指出,ref比reactive更适用。下面我们从源码的角度详细讨论这两个API,以及Vue3为什么推荐使用ref而不是reactive。
ref的内部工作原理是,它是一个函数,接受一个内部值并返回一个响应式且可变的引用对象。这个引用对象有一个.value属性,指向内部值。
在上述代码中,ref函数通过new RefImpl(value)创建了一个新的RefImpl实例。这个实例包含getter和setter,分别用于追踪依赖和触发更新。使用ref可以声明任何数据类型的响应式状态,包括对象和数组。
ref的核心是返回响应式且可变的引用对象,而reactive的核心是返回响应式代理,这是两者本质上的核心区别,也就导致了ref优于reactive。
reactive的内部工作原理是,它是一个函数,接受一个对象并返回该对象的响应式代理,也就是Proxy。
reactive的源码相对简单,通过new Proxy(target, baseHandlers)创建了一个代理。这个代理会拦截对目标对象的操作,从而实现响应式。
ref和reactive在声明数据的响应式状态上,底层原理不同。ref采用RefImpl对象实例,reactive采用Proxy代理对象。
当你使用new RefImpl(value)创建一个RefImpl实例时,这个实例大致上会包含以下几部分:Dep类负责管理一个依赖列表,并提供依赖收集和通知更新的功能。RefImpl类包含一个内部值_value和一个Dep实例。当value被访问时,通过get方法进行依赖收集;当value被赋予新值时,通过set方法触发更新。
尽管两者在内部实现上有所不同,但它们都能满足我们对于声明响应式变量的要求,但是reactive存在一定的局限性。
reactive的局限性包括仅对引用数据类型有效,使用不当会失去响应。reactive主要适用于对象,包括数组和一些集合类型(如Map和Set)。对于基础数据类型(如string、number和boolean),reactive是无效的。这意味着如果你尝试使用reactive来处理这些基础数据类型,将会得到一个非响应式的对象。
ref()为响应式编程提供了一种统一的解决方案,适用于所有类型的数据,包括基本数据类型和复杂对象。以下是推荐使用ref的几个关键原因:统一性、深层响应性和灵活性。
ref的核心优势之一是它的统一性。它提供了一种简单、一致的方式来处理所有类型的数据,无论是数字、字符串、对象还是数组。这种统一性极大地简化了开发者的代码,减少了在不同数据类型之间切换时的复杂性。
ref支持深层响应性,这意味着它可以追踪和更新嵌套对象和数组中的变化。这种特性使得ref非常适合处理复杂的数据结构,如对象和数组。
ref提供了高度的灵活性,尤其在处理普通赋值和解构赋值方面。这种灵活性使得ref在开发中的使用更加方便,特别是在进行复杂的数据操作时。
ref在Vue3中提供了一种更统一、灵活的响应式解决方案,还能避免了reactive的某些局限性。希望这篇文章对你有所帮助,有所借鉴。