实现 effect 的 scheduler 功能

编写单测
我们先来看看测试样例
it('scheduler', () => {
// 1. scheduler 作为 effect 的一个 option
// 2. 有了 scheduler 之后原来的 fn 参数只会执行初始化的一次
// 3. 如果依赖更新时不会执行 fn ,而是会去执行 scheduler
// 4. runner 不受影响
let dummy
let run: any
const scheduler = vi.fn(() => {
run = runner
})
const obj = reactive({ foo: 1 })
// 在这里将 scheduler 作为一个 option 传入 effect
const runner = effect(
() => {
dummy = obj.foo
},
{ scheduler }
)
expect(scheduler).not.toHaveBeenCalled()
// 会执行一次 effect 传入的 fn
expect(dummy).toBe(1)
obj.foo++
// 有了 scheduler 之后,原来的 fn 就不会执行了
expect(scheduler).toHaveBeenCalledTimes(1)
expect(dummy).toBe(1)
run()
expect(dummy).toBe(2)
})
实现
class ReactiveEffect {
private _fn: any
// [scheduler] 构造函数加入 options,这里使用 public 可以供外部使用
constructor(fn, public options) {
this._fn = fn
}
// other code ...
}
// other code ...
export function trigger(target, key) {
const depsMap = targetMap.get(target)
const deps = depsMap.get(key)
for (const effect of deps) {
// [scheduler] 这里需要判断一下 scheduler,如果存在就去运行 scheduler 而不是 fn
if (effect.options.scheduler) {
effect.options.scheduler()
} else {
effect.run()
}
}
}
export function effect(fn, options: any = {}) {
// [scheduler]在创建 ReactiveEffect 实例的时候,保存一下 options
const _effect = new ReactiveEffect(fn, options)
// other code ...
}
这样我们再运行单元测试,就可以通过了
原文始发于微信公众号(WEB大前端):实现 effect 的 scheduler 功能
版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。
文章由极客之音整理,本文链接:https://www.bmabk.com/index.php/post/225215.html