目录
计算属性
使用微任务优化调度器
首页 web前端 Vue.js Vue3计算属性怎么实现

Vue3计算属性怎么实现

May 26, 2023 pm 06:36 PM
vue3

计算属性

Vue3的官方文档中,对于计算属性有这样的描述:

  • 对于任何包含响应式数据的复杂逻辑,我们都应该使用计算属性

  • 计算属性只会在相关响应式依赖发生改变时重新求值

从上面的描述可以明确计算属性的需求,计算属性计算的是响应式数据(满足描述1),且计算结果应当被缓存(满足描述2)。让我们一个一个来实现,先使用computed创建一个计算属性。

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

function effect(fn) { // 副作用函数

  const effectFn = () => {

    cleanup(effectFn)

    activeEffect = effectFn

    effectStack.push(effectFn)

    fn()

    effectStack.pop()

    activeEffect = effectStack[effectStack.length - 1]

  }

  effectFn.deps = []

  effectFn()

}

...

const data = { foo: 1, bar: 2 }

const obj = new Proxy(data, { // 响应式对象

  get(target, key) {

    track(target, key)

    return target[key]

  },

  set(target, key, newValue) {

    target[key] = newValue

    trigger(target, key)

    return true

  }

})

...

const sumRes = computed(() => obj.foo + obj.bar) // (1)

console.log(sumRes.value)

登录后复制

在(1)处,我们简单写了一个计算属性的功能,为了实现通过sumRes.value读取计算属性值功能,在实现计算属性时,需要返回一个对象,通过对象内的get触发副作用函数。

1

2

3

4

5

6

7

8

9

function computed(getter) {

  const effectFn = effect(getter)

  const obj = {

    get value() {

      return effectFn()

    }

  }

  return obj

}

登录后复制

但这个函数显然是无法执行的,这是因为前面我们在实现effect时,需要直接执行副作用函数,不需要提供返回值。没有返回值,computed自然无法获取到effect的执行结果。因此,当在计算属性中使用effect时,需要将副作用函数返回给计算属性,由计算属性决定何时执行,而不再由effect立即执行(即懒执行)。

为了实现这点,就需要向effect中添加一个开关lazy,考虑到我们可能将来还需要对effect配置其它特性,我们使用一个对象options来封装这个开关。

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

function effect(fn, options = {}) {

  const effectFn = () => {

    cleanup(effectFn)

    activeEffect = effectFn

    effectStack.push(effectFn)

    const res = fn() // (1)

    effectStack.pop()

    activeEffect = effectStack[effectStack.length - 1]

    return res // (2)

  }

  effectFn.deps = []

  effectFn.options = options // (3)

  if (!options.lazy) { // (4)

    effectFn()

  }

  return effectFn // (5)

}

登录后复制

我们在(4)处放置了lazy开关,不需要懒执行的副作用函数同样会自动执行。在(1)(2)(5)处返回了副作用函数的结果,供懒执行使用。同时在(3)处向下传递了options,保证在effect发生嵌套时,也使得副作用函数执行预期的行为。基于上述effect的修改,我们在computed中设置lazy开关。

1

2

3

4

5

6

7

8

9

10

function computed(getter) {

  const effectFn = effect(getter, { lazy: true })

  const obj = {

    get value() { // (6)

      return effectFn()

    }

  }

  return obj

}

const sumRes = computed(() => obj.foo + obj.bar)

登录后复制

Vue3计算属性怎么实现

从上图中可以看出,我们已经实现了描述1,即使用计算属性进行响应式数据的计算,当响应式数据的值发生变化时,计算属性的值也会随之改变。但观察上文代码的(6)处,不难发现,无论什么情况下,只要读取sumRes.value的值,就会触发一次副作用函数,使其重新进行可能不必要的执行。所以接着,我们尝试实现描述2,缓存计算属性的结果。

先从最简单的入手,我们用一个变量value来缓存上次计算的值,并添加一个dirty开关,记录是否需要重新触发副作用函数。

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

function computed(getter) {

  let value

  let dirty = true

  const effectFn = effect(getter, { lazy: true })

  const obj = {

    get value() {

      if(dirty) {

        value = effectFn()

        dirty = false

      }

      return value

    }

  }

  return obj

}

登录后复制

修改之后,缓存的值就能生效了。但这样做产生了一个明显的BUG,当dirty的值被置为false时,无法再变为true,这也就意味着,无论响应式数据obj.barobj.foo如何变化,计算属性的值永远都只能是缓存的值value,如下图所示。

Vue3计算属性怎么实现

为了解决这个问题,我们需要一种方式,能够在obj.barobj.foo的值变化时,在获取sumRes.value之前,将dirty开关的值置为true。受前面懒加载的启发,我们尝试能不能通过配置options来实现这个功能。

1

2

3

4

5

6

7

8

9

10

11

const obj = new Proxy(data, {

  get(target, key) {

    track(target, key)

    return target[key]

  },

  set(target, key, newValue) {

    target[key] = newValue

    trigger(target, key)

    return true

  }

})

登录后复制

再来回忆一下响应式对象的整个流程,当响应式对象中的数据被修改时,执行了trigger去触发收集的副作用函数。而在计算属性中,我们不再需要自动的触发副作用函数。所以自然会想到,能否在这个地方将dirty置为true呢?按照这个思路,我们先对trigger进行修改。

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

function trigger(target, key) {

  const propsMap = objsMap.get(target)

  if(!propsMap) return

  const fns = propsMap.get(key)

  const otherFns = new Set()

  fns && fns.forEach(fn => {

    if(fn !== activeEffect) {

      otherFns.add(fn)

    }

  })

  otherFns.forEach(fn => {

    if(fn.options.scheduler) { // (7)

      fn.options.scheduler()

    } else {

      fn()

    }

  })

}

登录后复制

按照前文的思路,我们在(7)处增加了一个判断,如果副作用函数fn的配置项options中含有scheduler函数,我们就执行scheduler而非副作用函数fn。我们称这里的scheduler调度器。相应的,我们在计算属性中添加调度器。

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

function computed(getter) {

  let value

  let dirty = true

  const effectFn = effect(getter, {

    lazy: true,

    scheduler() { // (8)

      dirty = true

    }

  })

  const obj = {

    get value() {

      if(dirty) {

        value = effectFn()

        dirty = false

      }

      return value

    }

  }

  return obj

}

登录后复制

在(8)处我们添加了调度器。添加调度器后,让我们再来串一下整个流程,当响应式数据被修改时,才会执行trigger函数。由于我们传入了调度器,因此trigger函数在执行时不再触发副作用函数,转而执行调度器,此时dirty开关的值变为了true。当程序再往下执行时,由于dirty已经变为true,副作用函数就可以正常被手动触发。效果如下图所示。

Vue3计算属性怎么实现

到这里为止,计算属性在功能上已经实现完毕了,让我们尝试完善它。在Vue中,当计算属性中的响应式数据被修改时,计算属性值会同步更改,进而重新渲染,并在页面上更新。渲染函数也会执行effect,为了说明问题,让我们使用effect简单的模拟一下。

1

2

const sumRes = computed(() => obj.foo + obj.bar)

effect(() => console.log('sumRes =', sumRes.value))

登录后复制

这里我们的预期是当obj.fooobj.bar改变时,effect会自动重新执行。这里存在的问题是,正常的effect嵌套可以被自动触发(这点我们在上一篇博客中已经实现了),但sumRes包含的effect仅会在被读取value时才会被get触发执行,这就导致响应式数据obj.fooobj.bar无法收集到外部的effect,收集不到的副作用函数,自然无法被自动触发。

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

function computed(getter) {

  let value

  let dirty = true

  const effectFn = effect(getter, {

    lazy: true,

    scheduler() {

      dirty = true

      trigger(obj, 'value') // (9)

    }

  })

  const obj = {

    get value() {

      if(dirty) {

        value = effectFn()

        dirty = false

      }

      track(obj, 'value') // (10)

      return value

    }

  }

  return obj

}

登录后复制

在(10)处我们手动收集了副作用函数,并当响应式数据被修改时,触发它们。

Vue3计算属性怎么实现

使用微任务优化调度器

在设计调度器时,我们忽略了一个潜在的问题。

还是先来看一个例子:

1

2

3

4

effect(() => console.log(obj.foo))

for(let i = 0; i < 1e5; i++) {

  obj.foo++

}

登录后复制

Vue3计算属性怎么实现

随着响应式数据obj.foo被不停修改,副作用函数也被不断重复执行,在明显的延迟之后,控制台打印出了大量的数据。但在前端的实际开发中,我们往往只关心最终结果,拿到结果显示在页面上。在这种条件下,我们如何避免副作用函数被重复执行呢?

1

2

3

4

5

6

7

8

9

10

11

12

13

14

const jobQueue = new Set() // (11)

const p = Promise.resolve() // (12)

let isFlushing = false // (13)

function flushJob() { // (14)

  if (isFlushing) return

  isFlushing = true

  p.then(() => {

    jobQueue.forEach(job => {

      job()

    })

  }).finally(() => {

    isFlushing = false

  })

}

登录后复制

这里我们的思路是使用微任务队列来进行优化。(11)处我们定义了一个Set作为任务队列,(12)处我们定义了一个Promise来使用微任务。精彩的部分来了,我们的思路是将副作用函数放入任务队列中,由于任务队列是基于Set实现的,因此,重复的副作用函数仅会保留一个,这是第一点。接着,我们执行flushJob(),这里我们巧妙的设置了一个isFlushing开关,这个开关保证了被微任务包裹的任务队列在一次事件循环中只会执行一次,而执行的这一次会在宏任务完成之后触发任务队列中所有不重复的副作用函数,从而直接拿到最终结果,这是第二点。按照这个思路,我们在effect中添加调度器。

1

2

3

4

5

6

effect(() => { console.log(obj.foo) }, {

  scheduler(fn) {

    jobQueue.add(fn)

    flushJob()

  }

})

登录后复制

Vue3计算属性怎么实现

需要注意的是,浏览器在执行微任务时,会依次处理微任务队列中的所有微任务。因此,微任务在执行时会阻塞页面的渲染。因此,在实践中需避免在微任务队列中放置过于繁重的任务,以免导致性能问题。

以上是Vue3计算属性怎么实现的详细内容。更多信息请关注PHP中文网其他相关文章!

本站声明
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

热AI工具

Undresser.AI Undress

Undresser.AI Undress

人工智能驱动的应用程序,用于创建逼真的裸体照片

AI Clothes Remover

AI Clothes Remover

用于从照片中去除衣服的在线人工智能工具。

Undress AI Tool

Undress AI Tool

免费脱衣服图片

Clothoff.io

Clothoff.io

AI脱衣机

Video Face Swap

Video Face Swap

使用我们完全免费的人工智能换脸工具轻松在任何视频中换脸!

热门文章

<🎜>:泡泡胶模拟器无穷大 - 如何获取和使用皇家钥匙
3 周前 By 尊渡假赌尊渡假赌尊渡假赌
Mandragora:巫婆树的耳语 - 如何解锁抓钩
3 周前 By 尊渡假赌尊渡假赌尊渡假赌
北端:融合系统,解释
3 周前 By 尊渡假赌尊渡假赌尊渡假赌

热工具

记事本++7.3.1

记事本++7.3.1

好用且免费的代码编辑器

SublimeText3汉化版

SublimeText3汉化版

中文版,非常好用

禅工作室 13.0.1

禅工作室 13.0.1

功能强大的PHP集成开发环境

Dreamweaver CS6

Dreamweaver CS6

视觉化网页开发工具

SublimeText3 Mac版

SublimeText3 Mac版

神级代码编辑软件(SublimeText3)

热门话题

Java教程
1669
14
CakePHP 教程
1428
52
Laravel 教程
1329
25
PHP教程
1273
29
C# 教程
1256
24
Vue3如何实现刷新页面局部内容 Vue3如何实现刷新页面局部内容 May 26, 2023 pm 05:31 PM

想要实现页面的局部刷新,我们只需要实现局部组件(dom)的重新渲染。在Vue中,想要实现这一效果最简便的方式方法就是使用v-if指令。在Vue2中我们除了使用v-if指令让局部dom的重新渲染,也可以新建一个空白组件,需要刷新局部页面时跳转至这个空白组件页面,然后在空白组件内的beforeRouteEnter守卫中又跳转回原来的页面。如下图所示,如何在Vue3.X中实现点击刷新按钮实现红框范围内的dom重新加载,并展示对应的加载状态。由于Vue3.X中scriptsetup语法中组件内守卫只有o

vue3项目中怎么使用tinymce vue3项目中怎么使用tinymce May 19, 2023 pm 08:40 PM

tinymce是一个功能齐全的富文本编辑器插件,但在vue中引入tinymce并不像别的Vue富文本插件一样那么顺利,tinymce本身并不适配Vue,还需要引入@tinymce/tinymce-vue,并且它是国外的富文本插件,没有通过中文版本,需要在其官网下载翻译包(可能需要翻墙)。1、安装相关依赖npminstalltinymce-Snpminstall@tinymce/tinymce-vue-S2、下载中文包3.引入皮肤和汉化包在项目public文件夹下新建tinymce文件夹,将下载的

vue3+vite:src使用require动态导入图片报错怎么解决 vue3+vite:src使用require动态导入图片报错怎么解决 May 21, 2023 pm 03:16 PM

vue3+vite:src使用require动态导入图片报错和解决方法vue3+vite动态的导入多张图片vue3如果使用的是typescript开发,就会出现require引入图片报错,requireisnotdefined不能像使用vue2这样imgUrl:require(’…/assets/test.png’)导入,是因为typescript不支持require所以用import导入,下面介绍如何解决:使用awaitimport

Vue3怎么解析markdown并实现代码高亮显示 Vue3怎么解析markdown并实现代码高亮显示 May 20, 2023 pm 04:16 PM

Vue实现博客前端,需要实现markdown的解析,如果有代码则需要实现代码的高亮。Vue的markdown解析库有很多,如markdown-it、vue-markdown-loader、marked、vue-markdown等。这些库都大同小异。这里选用的是marked,代码高亮的库选用的是highlight.js。具体实现步骤如下:一、安装依赖库在vue项目下打开命令窗口,并输入以下命令npminstallmarked-save//marked用于将markdown转换成htmlnpmins

Vue3复用组件怎么使用 Vue3复用组件怎么使用 May 20, 2023 pm 07:25 PM

前言无论是vue还是react,当遇到多处重复代码的时候,我们都会想着如何复用这些代码,而不是一个文件里充斥着一堆冗余代码。实际上,vue和react都可以通过抽组件的方式来达到复用,但如果遇到一些很小的代码片段,你又不想抽到另外一个文件的情况下,相比而言,react可以在相同文件里面声明对应的小组件,或者通过renderfunction来实现,如:constDemo:FC=({msg})=>{returndemomsgis{msg}}constApp:FC=()=>{return(

Vue3中怎么实现选取头像并裁剪 Vue3中怎么实现选取头像并裁剪 May 29, 2023 am 10:22 AM

最终效果安装VueCropper组件yarnaddvue-cropper@next上面的安装值针对Vue3的,如果时Vue2或者想使用其他的方式引用,请访问它的npm官方地址:官方教程。在组件中引用使用时也很简单,只需要引入对应的组件和它的样式文件,我这里没有在全局引用,只在我的组件文件中引入import{userInfoByRequest}from'../js/api'import{VueCropper}from'vue-cropper&

怎么使用vue3+ts+axios+pinia实现无感刷新 怎么使用vue3+ts+axios+pinia实现无感刷新 May 25, 2023 pm 03:37 PM

vue3+ts+axios+pinia实现无感刷新1.先在项目中下载aiXos和pinianpmipinia--savenpminstallaxios--save2.封装axios请求-----下载js-cookienpmiJS-cookie-s//引入aixosimporttype{AxiosRequestConfig,AxiosResponse}from"axios";importaxiosfrom'axios';import{ElMess

vue3项目打包发布到服务器后访问页面显示空白怎么解决 vue3项目打包发布到服务器后访问页面显示空白怎么解决 May 17, 2023 am 08:19 AM

vue3项目打包发布到服务器后访问页面显示空白1、处理vue.config.js文件中的publicPath处理如下:const{defineConfig}=require('@vue/cli-service')module.exports=defineConfig({publicPath:process.env.NODE_ENV==='production'?'./':'/&

See all articles