实现 effect 的 scheduler 功能

实现 effect 的 scheduler 功能

实现 effect 的 scheduler 功能
实现 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

(0)
小半的头像小半

相关推荐

发表回复

登录后才能评论
极客之音——专业性很强的中文编程技术网站,欢迎收藏到浏览器,订阅我们!