对于@vue/reactivity相关的Api,比如ref、computed、reactive、effect、watch等,根据当前的执行环境是否在组件上下文中,有以下两种情况:
// 组件实例被创建的时候也会创建一个scope
export function createComponentInstance(...args) {// ...const instance = {// ...vnode,type,scope: new EffectScope(true /* detached */),// ...}return instance
}
// 组件卸载时会调用 stop 方法
const unmountComponent = (...args) => {// ...scope.stop()// ...
}// watchEffect 和 watch 返回的 stop 方法
function doWatch() {// ...return () => {effect.stop()if (instance && instance.scope) {remove(instance.scope.effects!, effect)}}
}
//(参考 vue-RFC 示例代码)
const disposables = []const counter = ref(0)
const doubled = computed(() => counter.value * 2)//需要手动消除依赖
disposables.push(() => stop(doubled.effect))const stopWatch1 = watchEffect(() => {console.log(`counter: ${counter.value}`)
})disposables.push(stopWatch1)const stopWatch2 = watch(doubled, () => {console.log(doubled.value)
})disposables.push(stopWatch2)
// 使用到的组件都会重复创建监听器
function useMouse() {const x = ref(0)const y = ref(0)function handler(e) {x.value = e.xy.value = e.y}window.addEventListener('mousemove', handler)onUnmounted(() => {window.removeEventListener('mousemove', handler)})return { x, y }
}
function useMouse() {const x = ref(0)const y = ref(0)function handler(e) {x.value = e.xy.value = e.y}window.addEventListener('mousemove', handler)// 通过onScopeDispose替换onUnmounted,意味着可以脱离组件使用onScopeDispose(() => {window.removeEventListener('mousemove', handler)})return { x, y }
}function createSharedComposable(composable) {let subscribers = 0let state, scopeconst dispose = () => {// 通过闭包进行计数,当subscribers为0时,stop掉该scope// 如果在组件中使用,则onUnmounted就意味着subscribers-1if (scope && --subscribers <= 0) {scope.stop()state = scope = null}}return (...args) => {subscribers++if (!state) {scope = effectScope(true)state = scope.run(() => composable(...args))}onScopeDispose(dispose)return state}
}const useSharedMouse = createSharedComposable(useMouse)export default useSharedMouse
// useGlobalState
import { effectScope } from '@vue/composition-api'export default run => {let stateconst scope = effectScope(true)return () => {// 防止重复触发if (!state) {state = scope.run(run)}return state}
}// store.js
import { computed, ref } from '@vue/composition-api'
import useGlobalState from './useGlobalState'export default useGlobalState(() => {// stateconst count = ref(0)// gettersconst doubleCount = computed(() => count.value * 2)// actionsfunction increment() {count.value++}return { count, doubleCount, increment }}
)