信仰崩了?Preact 开始采用 Vue3 的响应式设计

news2025/1/11 20:40:48

前言

不知大家有没有听过Preact这个框架,就算没听过Preact那也应该听过React吧?

一字之差,preactreact多了个p!(听起来咋不像啥好话呢)

这个P代表的是 Performance高性能版React的意思。Preact一开始是CodePen上的一个小项目,有点类似于咱们国内常见的《三百行代码带你实现个React》这类文章,用最少的代码实现了React的最基本功能,然后放到CodePen上供大家敬仰学习。

当然这是很多年前的事了,那时候这种东西很容易火,想想N年前你看过的第一篇《三百行实现个Vue》《三百行实现个React》之类的文章是不是竞争对手很少、很容易引发大量的关注度。不过现在不行了,太卷!这类文章隔三差五的就能看到一篇,同质化严重,导致大家都有点审美疲劳了。

但在那个年代Preact就是这么火起来的,三百行实现了个React引发大量关注度之后,作者觉得自己做的这玩意好像还挺不错的哈!于是开始继续完善,完善后拿去一测试:性能简直完爆React呀!我这玩意不仅体积比你小、性能还比你高。就这样作者开始有些膨胀了、开始飘了!

那我就给这个框架起个名叫Preact吧!

Performance版的React

Preact 简介

打开Preact官网,映入眼帘的便是它的最大卖点:


只有3KB大小、并且与React拥有相同的API。真的只有3KB么?虚拟DOMDiff算法、类组件、Hooks… 这些就算实现的再怎么巧妙也需要很多代码才行吧?我们直接用Vite来创建一个Preact项目来试下:

npm create vite



如果屏幕前的你用的是VSCode这个编辑器的话,可以安装一下Import Cost这个插件:


安装好之后我们来看一下主文件(main.jsx):


卧槽?gizpped真的只有3.几K!不过这算法有点鸡贼啊,来了个向下取整:

这让我想起了最近非常火的TurbopackVite快十倍的宣传口号,遭尤大怒怼:1k 组件的案例下有数字的四舍五入问题,Turbopack15ms 被向下取整为 0.01s,而到了 Vite 这里 87ms 被向上取整为 0.09s。这把本来接近 6 倍的差距扩大到了 10 倍。

不过即使这样,3.8K依然是一个很惊人的成就。是不是只是render这个函数就占了3.8K啊?我们再引点东西试试:

难以置信!引了这么多hooks居然只多加了0.1K!我还是不太相信用0.1K的代码就能实现出React Hooks来,肯定是用了什么特殊的算法专门针对了这一场景做了优化,我们按照官网的写法来重新引一下:


这回体积明显增大了不少:


不过咋感觉自己跟个杠精似的呢😂 人家说了3KB我却非要以各种方式证明肯定不止3KB,这样不好。Preact真的已经很轻量了,一般人想要实现这么多功能还真做不到只用这么少的代码,PreactP肯定还是名不虚传的👍

不过刚刚试了下VueVue好像就没有针对这种场景做专门的优化,不仅没优化反而还劣化了:


实际上只引入某几个函数的话Vue没有这么大,这是把Vue全量引入的大小,尤大还不快跟人学学。

Preact Signals

说到"学"Preact原本一直都是React的忠实粉丝,可最近它却开发了一个叫做@preact/signals的东西,这是干嘛的?Preact的创始人Jason Miller以及Preact DevTools的创始人Marvin Hagemeister共同写了篇博客:《Introducing Signals》

点开文章,首先映入眼帘的便是这样一个案例:

import { signal, computed } from "@preact/signals";

const count = signal(0);
const double = computed(() => count.value * 2);

function Counter() {
  return (
    <button onClick={() => count.value++}>
      {count} x 2 = {double}
    </button>
  );
}

等等!这个.value、这个computed、以及这个在jsx的大括号{}中不用写.value的语法…

怎么这么似曾相识呢?好像在哪里见过类似的写法:

<script setup>
import { ref, computed } from "vue";

const count = ref(0);
const double = computed(() => count.value * 2);
</script>

<template>
  <button @click="count++">
    {{count}} x 2 = {{double}}
  </button>
</template>

hooks不同,signals可以在组件内部或外部使用。signals在类组件也可以很好的运行,因此您可以按照自己的节奏引入它们,并根据现状,在几个组件中试用它们,并随着时间的推移逐渐采用它们。—— Preact团队

那这样不是越写越Vue了吗?还叫什么Preact啊,叫Vreact多好!

尤雨溪:这还真特娘的是个好主意!我这就把拉你进 Vue 核心群里来!

我们来看看Preact团队为何要实现个P版的Composition API

  • 易冲突的全局状态
  • 混乱的上下文
  • 寻求更好的状态管理
  • 卓越的性能

听说最近尤大被骂了,为啥被骂呢?因为好像有次字节邀请了尤大直播,那尤大肯定得借此机会好好宣传一下Vue啊!不过你光说Vue有多好,观众可能无法感受到。就像如果七龙珠直接让超级赛亚人出场,并且用那个战斗力探测仪显示一个战力:

虽说凭这个确实能让人感受到超级赛亚人很强,但如果要是能有个对比的话那才是最完美的剧情,所以才有了大反派弗利萨的出场机会:

同理,尤大如果光在那罗列数据那肯定不如有个对比来的直观,那就把React拉来对比一番呗!既然是为了宣传Vue,那必须得拿Vue的优点跟React的缺点比啦!这样的对比难免会有失偏颇,让React的粉丝们怒不可遏,在群里疯狂批判尤大。

在一捧一踩(黑React)的宣传过程中呢,尤大花费最多时间宣传的就是以下两点:

  • 避免了React Hooks的一些心智负担
  • 性能比React

其实这两点多少还是有点有失偏颇,因为Vue在解决了一种心智负担的同时又带来了另一种心智负担,而且性能也要看场景的,尤大只强调了对Vue有利的场景来宣传…

不过React在某些层面来讲确实有些剑走偏锋了哈,导致性能不是特别理想。Preact也是这么认为的,他们还特意搞了张火焰图:


左边用的是Preact Hooks,右边用的是Composition API… 哦不,是Preact Signals。可以看到Signals的表现完胜Hooks

Preact的老师React有在React里实现Vue的计划吗?答案是否定的,自从Preact Signals发布后大家就疯狂@DanDan看完后直接来了句:这与React的发展理念不是很吻合。(潜台词:我们才不会在React里实现Vue呢)

其实我觉得也是,React的发展理念本来就跟Vue走的是完全不同的两种路线,夸张点说就是道不同不相为谋。

那肯定有人说:不对呀,Vue3Composition API不是抄袭的React么?

这么说吧:大佬们借鉴的是思路,菜鸟们借鉴的才是代码。了解过VueReact他俩底层实现的朋友们应该都清楚他俩的差距有多大,尤雨溪在某次采访时说过Vue3一开始本打算实现成类组件,既然是类那就离不开装饰器的话题,尤大说他们甚至都已经实现出来了一版类组件写法的Vue3。只不过他觉得这样相对于Vue2而言除了对TS的支持度之外几乎没有其他什么特别明显的优势,而且装饰器提案发展了N年却迟迟未能落地,尤大觉得这样遥遥无期,而且就算真的在将来的某一天落地了,是不是也已经与现在TS实现的那版装饰器天差地别了?

Angular用装饰器用的好好的那是因为人家强制要求使用TS,但Vue显然不可能这样做。而且为了防止未来装饰器有变动(其实最近已经Stage3的装饰器已经和TS装饰器不一样了),许多曾经使用装饰器语法的库为了规避这个风险也已经改用了别的写法,如:MobXReact DnD等…

推荐阅读:《mobx6.0为什么移除装饰器》

正当尤雨溪为此抓耳挠腮、夜不能寐之时,React Hooks横空出世了!这种函数式组件瞬间就让尤大眼前一亮,他脑袋里的灯泡在那一刹那间被点亮了:


这不就是自己一直苦苦寻找、对TS友好、方便代码复用、语法简洁、低耦合的解决方案么!

但实际上吧,尤大只是参考了这种函数式的设计,如今的Composition API原理与React Hooks相去甚远。真要说借鉴的话,尤雨溪已经大大方方承认了是受到了React Hooks的启发,代码层面借鉴的是Meteor Tracker、nx-js/observer-util、salesforce/observable-membrane这三个库。响应式库其实早已不新鲜了,只是之前尤大没能跳出Vue2的思维限制,直到看到了React Hooks才想到可以这样写,然后再一调研发现市面上早就有了函数式的响应式库,Composition API就是这么来的。

推荐阅读:《[译]尤雨溪:Vue3的设计过程》

不过他在Composition API之前确实模仿了React的原理设计出来了vue-hooks,以用来探索这种函数式组件的可行性。不过好在后来他发现了Meteor Tracker、nx-js/observer-util、salesforce/observable-membrane这几个库并及时悬崖勒马,没有在这个方向上继续深挖,不然的话Vue3可能就要变成套壳React了。

那究竟为什么没有在此方向继续深挖呢?难道说那仨库的解决方案比React Hooks还要好吗?对此我只想说:

抛开了场景谈好坏都是在耍流氓

这两种方案各有优缺点,巧合的是:双方彼此间的优点恰恰好好就是对方身上的缺点。典型的性格互补么这不是:

有人喜欢内向的、有人喜欢外向的、但也有人想当一个缝合怪:为啥不能内外双向呢?该内向的时候就内向,该外向的时候就外向呗!Preact就是这样想的,他们单独提供了一个叫@preact/signals的包,你要是更在意性能呢,那就用@preact/signals、你要是更在意类似React的开发体验呢,那就不用呗!

用法

Preact版的composition api主要分为三个部分:

  • @preact/signals-core
  • @preact/signals
  • @preact/signals-react

从命名上来看,@preact/signals-core应该是与框架无关的核心实现、@preact/signals是给Preact的特供产品、而@preact/signals-react则是给React提供的特供产品。

我们先来看一下核心实现的用法,这是他们README文件里给出的第一个例子:

import { signal } from "@preact/signals-core";

const counter = signal(0);

// Read value from signal, logs: 0
console.log(counter.value);

// Write to a signal
counter.value = 1;

非常好理解,就是把原来composition api里的ref换成了signal,这里就不过多赘述了,来看下一个案例:

const counter = signal(0);
const effectCount = signal(0);

effect(() => {
  console.log(counter.value);

  // Whenever this effect is triggered, increase `effectCount`.
  // But we don't want this signal to react to `effectCount`
  effectCount.value = effectCount.peek() + 1;
});

这个effect也和composition api里的effect如出一辙,不过有同学可能会问了:composition api里没有effect呀?你说的是watchEffect吗?我这里表述的可能不是特别准确,准确来讲的话应该是和@vue/reactivity里的effect如出一辙。

那么问题来了:@vue/reactivity不就是composition api吗?其实他俩确实非常的…容易混淆,准确来讲@vue/reactivity是可以运行在完全脱离vue的环境之下的,而composition api是根据vue的环境进行的进一步更好用的封装。composition api包含了@vue/reactivity

composition api@vue/composition-api又有啥区别呢?区别就是composition api只是一个概念,而@vue/composition-api是一个实现了composition api的项目。当初尤雨溪提出composition api的时候(那时候还不叫composition api,好像叫什么functional base api)遭到了大量质疑的声音,于是有个大佬就用Vue2现有的API实现了一版尤雨溪的提案,尤雨溪觉得这玩意非常不错!你们老喷我是因为你们没有体验过函数式的好,你们先用用试试,试完了保证你们直呼真香!于是联系该作者把Vue2版的composition api合并到Vue的仓库中并发布为@vue/composition-api。但谁也不会用爱发电对不,刚开始当个娱乐项目给你宣传了,时间一长也没啥收益,该作者也就不维护了。此时另一位大佬出现了,他说既然没人维护了那就交给我吧!他就是肝帝AntFu


一整年就两三天是灭着的,剩下的时间无论刮风还是下雨,都无法阻挡大佬提交代码的脚步。甚至那两三天我都怀疑是有什么不可抗力导致的,比方说来台风断电啦或者在飞机上没法提交,下了飞机直接就进入另一个时区(第二天)啦之类的原因,他甚至比尤雨溪都勤快:

不过拿他俩比有点不太公平哈,尤大有家有孩子,而且还要带领两个团队(VueVite),写代码的时间自然会少很多。而佬年轻没结婚没孩子、也无需带领团队啥的,自然就会有很多时间做自己喜欢做的事情。不过我翻了一下尤大迭代最疯狂的2016年,也依然没我傅哥勤快:

这就是我傅哥为何能如此高产的原因。

有点扯远了哈,没接触过@vue/reactivityeffect同学暂且先把它理解为composition apiwatchEffect,在这里开始出现了一个与composition api不太一样的api了哈,.peek()是什么鬼?为了帮助大家快速理解这玩意,我们需要对比一下composition api里两个相似功能的apiwatchwatchEffect

这俩api功能相似但各有优缺点,我们只说watchEffect不如watch的其中一个缺点:无法精确控制到底监听了哪个响应式变量。

比方说我们写了这样一段逻辑:

import { ref, watchEffect } from 'vue';

const a = ref(0);
const b = ref(0);

watchEffect(() => {
  console.log(a.value);

  b.value++;
});

每当我们改动a.value的值时,b.value就会++。这是我们希望的逻辑,但不幸的是,每当我们改动b.value的值时,b.value还是会++。这在watch里还是很好实现的:

import { ref, watch } from 'vue';

const a = ref(0);
const b = ref(0);

watch(a, value => {
  console.log(value);

  b.value++;
});

但在watchEffect那段代码里就相当于在watch中写了这样一段代码:

import { ref, watch } from 'vue';

const a = ref(0);
const b = ref(0);

watch([a, b], ([valueA, valueB]) => {
  console.log(valueA);

  b.value = valueB + 1;
});

Vue的方案是既提供一个自动收集依赖的watchEffect,同时也提供一个手动收集依赖的watch

Preact的方案则是只提供一个effect(类似VuewatchEffect),如果你写出类似上面那样的代码:

import { signal, effect } from '@preact/signals-core';

const a = signal(0);
const b = signal(0);

effect(() => {
  console.log(a.value);

  b.value++;
});

那就直接报错给你看:

为什么会报错呢?了解过响应式原理的同学应该不难理解,就是触发getter的时候又会触发setter,而触发了setter又会导致重新运行effect函数导致死循环。但如果你没了解过响应式原理的话可能就不太清楚我说的到底是什么意思,建议阅读一下这篇:

《尤大:怎么还生啃源码呢?我这就亲手给你写个丐版Vue》

这是尤大在国外的VueMastery教程网站中直播的手写简易版Vue,文章里有详细的图例来帮助大家快速理解,短短几十行代码就能实现一个简易的响应式系统,吃透了原理你就会明白为什么这样写会导致死循环了。

那为啥Vue那边的代码没死循环呢?这是因为Vue做了这样一层判断:如果你在effect / watchEffect里触发了setter,那便不会触发对应的effect / watchEffect函数,这样就可以避免死循环了。

Preact没做这样的处理怎么办呢?那我们就避免在effect里既对signal进行取值操作同时又对它进行赋值操作呗!

不过这样做肯定是不行的哈,你这太不专业了,所有成熟的响应式库没有哪个会放着这个问题不去解决的。比方说Solid.js,他就有一个叫untrack的函数,假如我们在effect里想要获取到一个响应式的值但却并不想它被收集到依赖里面去就可以写成这样:

import { createSignal, createEffect, untrack } from "solid-js";

const [a, setA] = createSignal(0);
const [b, setB] = createSignal(0);

createEffect(() => {
  console.log(a());
  console.log(untrack(b));
});

这样只有a改变时会触发effect函数,b则不会。如果你能理解上面这段代码的话,那相信你肯定能理解下面这段代码:

import { signal, effect } from '@preact/signals-core';

const a = signal(0);
const b = signal(0);

effect(() => {
  console.log(a.value);

  b.value = b.peek() + 1;
});

我还专门去查了一下peek是啥意思,是偷窥的意思。有时候觉得老外起的api名翻译过来还蛮有意思的,就是说我在effect里需要获取到某个响应式变量的值,但直接获取会被追踪到,所以我不直接获取,我要“偷窥”一眼它的值,这样就不会被追踪到啦!(这个api虽然很调皮,但有些略显猥琐)



接下来看下一个apicomputed。就我不说你们都能猜到这是干啥的,这就是Vue的那个computed,直接看例子就不解释了:

import { signal, computed } from "@preact/signals-core";

const name = signal("Jane");
const surname = signal("Doe");

const fullName = computed(() => name.value + " " + surname.value);

// Logs: "Jane Doe"
console.log(fullName.value);

// Updates flow through computed, but only if someone
// subscribes to it. More on that later.
name.value = "John";
// Logs: "John Doe"
console.log(fullName.value);

下一个apieffect,其实在.peek()那个“偷窥”案例中就已经用过effect了,它就是@vue/reactivity里的effect,也不过多解释了,直接上案例:

import { signal, computed, effect } from "@preact/signals-core";

const name = signal("Jane");
const surname = signal("Doe");
const fullName = computed(() => name.value + " " + surname.value);

// Logs: "Jane Doe"
effect(() => console.log(fullName.value));

// Updating one of its dependencies will automatically trigger
// the effect above, and will print "John Doe" to the console.
name.value = "John";
import { signal, computed, effect } from "@preact/signals-core";

const name = signal("Jane");
const surname = signal("Doe");
const fullName = computed(() => name.value + " " + surname.value);

// Logs: "Jane Doe"
const dispose = effect(() => console.log(fullName.value));

// Destroy effect and subscriptions
dispose();

// Update does nothing, because no one is subscribed anymore.
// Even the computed `fullName` signal won't change, because it knows
// that no one listens to it.
surname.value = "Doe 2";

接下来这个api可能会有些令大家陌生了,叫batch,分批处理的意思,来看如下案例:

import { signal, computed, effect, batch } from "@preact/signals-core";

const name = signal("Jane");
const surname = signal("Doe");
const fullName = computed(() => name.value + " " + surname.value);

// Logs: "Jane Doe"
effect(() => console.log(fullName.value));

// Combines both signal writes into one update. Once the callback
// returns the `effect` will trigger and we'll log "Foo Bar"
batch(() => {
  name.value = "Foo";
  surname.value = "Bar";
});

有一定开发经验的同学应该一下子就能看出这段代码想表达什么意思了(如果看不懂的话去反思一下),就是当我们修改值的时候是同步出发对应的effect函数的,所以我们如果连着改两次就会连续运行两次,我写了一个简化版的案例给大家看一下:

import { signal, effect } from "@preact/signals-core";

const a = signal(0);

effect(() => console.log(a.value));

a.value++;
a.value++;

控制台打印结果:

就挺让人无语的…… 这也能水个API出来?人家Vue默认就是分批处理的,我们在Vue里写一段同样的代码来看看Vue是怎么运行的:

import { ref,  watchEffect } from 'vue';

const a = ref(0);

watchEffect(() => console.log(a.value));

a.value++;
a.value++;

控制台打印结果:

不过之前咱们不是说Vue的响应式依赖是@vue/reactivity么?Composition APIVue@vue/reactivity的基础上再次封装,让它变得更好用更适合Vue项目。那会不会是它封装了批处理才导致这样的结果的呢?我们先不用import xxx from 'vue'这种形式了,这样的话用的是Composition API,我们这次用@vue/reactivity再来试一把:

import { ref,  effect } from '@vue/reactivity';

const a = ref(0);

effect(() => console.log(a.value));

a.value++;
a.value++;

果不其然,这次的结果终于和Preact保持一致了:

误会了哈!我还寻思@preact/signals-core也太不专业了,人家@vue/reactivity默认就支持的东西……

不过既然@vue/reactivity默认也是同步的,那怎么分批处理呢?想让它像@preact/signals-core这样:

import { signal, effect, batch } from "@preact/signals-core";

const a = signal(0);

effect(() => console.log(a.value));

batch(() => {
  a.value++;
  a.value++;
})

@vue/reactivity中要想要达到同样效果的话… 关键是这个@vue/reactivity连个文档都没有!Vue官网上的Composition API是又封装了一层,用法已经不一样了。比方说Composition API里的watch@vue/reactivity里就没有,而且watchEffecteffect表现也不太一致,@vue/reactivityREADME写的也特别简陋:


机翻一下:


就很无奈,我想知道这个库怎么用就只能去看看它的TS定义,看看都有哪些API以及都有哪些用法。哪怕不像Vue那样有个专门的官网,那你在README里写几个简单的事例也行啊!就像@preact/signals-core那样,能耽误你几小时?


吐槽归吐槽,想知道咋用还是得去看代码,在了src/effect.ts后我发现这样一段代码:

果然还是和Composition API里的watchEffect参数不一致,我们能看到有个lazy字段,从名字上来看应该就是它了吧。我还特意去Vue官网看了一下watchEffect的第二个参数都有哪些字段,watchEffect就没有lazy这个字段,取而代之的是flush字段:

用法这么大差异,连个文档都不写。尤大,你是想让每个用@vue/reactivity的人都去从源码里找答案么?算了不吐槽了,咱们继续来看例子:

import { ref, effect } from "@vue/reactivity";

const a = ref(0);

effect(() => console.log(a.value), { lazy: true });

a.value++;
a.value++;

加了{ lazy: true }以后控制台啥都不打印了!尤大你是要气死我呀!那这个lazy到底是用来干啥的?可能是用来代替Composition API里的watch的吧?wacth会自动执行一次,effect则不会这样。那也不对啊,watch只是刚开始的时候不会自动执行一次,但当依赖变化时还是会运行啊,这怎么连运行都不运行了?不是你别让我猜呀!想知道你这库咋用就两种方式:要么看源码要么就靠猜…… 那如果不是lazy的话那就是scheduler字段?想看看你这咋用,结果你给我来个这:


文档不写就算了,你还定义了一堆any类型… 这特么到底咋用啊?好像以前看过的《Vue.js设计与实现》里有写过,不过那本书搬家放在哪里想不起来了,等我找到后再把例子给补上。

之前还想吐槽@preact/signals-core不专业,Vue早就支持的功能它还要专门出一个API。现在看来还是我太年轻,与框架无关的@vue/reactivity连个文档都没有,都不知道怎么支持这个批量更新,不专业的反而是@vue/reactivity

咱们继续来看下一个案例:

import { signal, computed, effect, batch } from "@preact/signals-core";

const counter = signal(0);
const double = computed(() => counter.value * 2);
const tripple = computed(() => counter.value * 3);

effect(() => console.log(double.value, tripple.value));

batch(() => {
  counter.value = 1;
  // Logs: 2, despite being inside batch, but `tripple`
  // will only update once the callback is complete
  console.log(double.value);
});
// Now we reached the end of the batch and call the effect

这是啥意思呢?就是我们在batch函数里访问了一个计算属性,按理说要等batch函数运行完了才会去更新,但这个计算属性依赖的值在batch里刚刚被改过,为了让我们能拿到正确的值,不等batch执行完就直接更新这个计算属性。但也不是所有依赖counter的计算属性都会被更新,没在batch函数里被访问到的tripple就会等batch函数运行完毕后再去进行更新。

batch函数还可以嵌套着写:

import { signal, computed, effect, batch } from "@preact/signals-core";

const counter = signal(0);
effect(() => console.log(counter.value));

batch(() => {
  batch(() => {
    // Signal is invalidated, but update is not flushed because
    // we're still inside another batch
    counter.value = 1;
  });

  // Still not updated...
});
// Now the callback completed and we'll trigger the effect.

当最外层的batch函数运行完成时才会更新对应的值。

React 及 Preact

core的核心部分讲完了,那就继续看看@preact/signals以及@preact/signals-react吧!它俩用法都一样:

import { signal } from "@preact/signals-react";

const count = signal(0);

function CounterValue() {
  // Whenver the `count` signal is updated, we'll
  // re-render this component automatically for you
  return <p>Value: {count.value}</p>;
}
import { useSignal, useComputed } from "@preact/signals-react";

function Counter() {
  const count = useSignal(0);
  const double = useComputed(() => count.value * 2);

 return (
    <button onClick={() => count.value++}>
      Value: {count.value}, value x 2 = {double.value}
    </button>
  );
}

就有点类似于在React里写Vue的那种感觉。

后续

晚上回家一顿翻,终于找着了《Vue.js设计与实现》这本书,声明一下本文真不是这本书的软广告,多卖出去一本我也不会得到什么分成。真就是我写那个例子的时候找不到文档又不知道咋用,README让去看TS声明结果看了个any

我是真没耐心去特别仔细的研究@vue/reactivity的源码,我觉得理解了大概的原理就行不必那么死抠细节,毕竟咱们一不靠卖源码课赚钱、二也不负责维护Vue、三也不像一些大佬似的没事就以钻研为乐、四也不至于研究完源码就能升职加薪什么的…

不过好在我之前看过那本书里面写的挺详细的好像有schedulerlazy之类的字段是用来干嘛的并且还给出了实现以及用例。我又看了一遍响应式那章,之前靠猜以为lazy是用来模仿watch的,结果写了{ lazy: true }之后直接不运行了,这是因为写了{ lazy: true }就从自动挡变手动挡了!返回一个函数让你自己去决定啥时候运行:

import { ref, effect } from '@vue/reactivity'

const a = ref(0)
const fn = effect(() => console.log(a.value), { lazy: true })

a.value++
fn()

打印结果:

那这样写有什么意义呢?这样写确实没什么意义,本来能自动运行的函数非要让你手动运行。这样做的意义主要是为了实现computed的,咱们想要的是@preact/signals-core里的batch批处理功能,书中的scheduler选项接收一个参数,但实测当前最新版本的@vue/reactvity没有任何参数:

import { ref, effect } from '@vue/reactivity'

const a = ref(0)
effect(
  () => console.log(a.value),
  {
    scheduler (fn) {
      console.log(fn)
    }
  }
)

a.value++

打印结果:

盲猜可能是版本变化导致的用法不一致行为,我们把@vue/reactivity的版本改成3.0后再来打印一下:


这回有值了,那为什么会把这个参数删掉呢?我们只能从CHANGELOG里找答案了:

从有限的信息我们可以得知大概是从3.2及后续版本删掉了的,3.0.x3.1.x与书中用法保持一致。在书中scheduler的参数十分重要,书中就是基于这个参数来实现的批处理能力。想知道新用法么?我不告诉你!就是不写文档嘿嘿!看源码去吧!


这让我突然想起尤大在某纪录片中吐槽有些人就是不看文档,我也想吐个槽:你特么倒是写呀!

没办法了,先钻研一下源码吧!经过我一段时间的钻研呢,大概得出来了一个结论:在3.2之后effect的返回值其实就相当于3.2之前scheduler的参数:

// 3.2 以前
effect(
  () => {},
  {
    scheduler (fn) {
      console.log(fn)
    }
  }
)

// 3.2 之后(含 3.2)
const fn = effect(
  () => {},
  {
    scheduler () {
      console.log(fn)
    }
  }
)

那我们就可以根据这一变化来重写书中给出的调度执行的案例了:

import { ref, effect } from '@vue/reactivity'

const jobQueue = new Set()
const p = Promise.resolve()

let isFlushing = false
function flushJob() {
  if (isFlushing) return

  isFlushing = true

  p.then(() => {
    jobQueue.forEach(job => job())
  }).finally(() => {
    isFlushing = false
  })
}

const a = ref(0)
const fn = effect(
  () => console.log(a.value),
  {
    scheduler () {
      jobQueue.add(fn)
      flushJob()
    }
  }
)

a.value++
a.value++

这次的打印结果就与@preact/signals-core保持一致了:

为什么会在3.2以后去掉这个这个参数呢?我觉得是因为这个参数与effect的返回值一致,相当于重复了,不信的话我们来拿3.0来做个实验:

import { ref, effect } from '@vue/reactivity'

const a = ref(0)
const fn = effect(
  () => console.log(a.value),
  {
    scheduler (func) {
      console.log(fn === func)
    }
  }
)

a.value++

打印结果:

吐槽:重复了你就在CHANGELOG里写一句因与返回值重复故删之类的话呗!啥也不写就非得让人去看源码

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/75149.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

一篇博客详解朴素贝叶斯解分类问题

目录 回归问题 正则化Regularization 分类问题—类别之间是独立的 为什么不用回归解分类问题以概率的形式解分类问题 抽盒机率→分类机率正态分布覆盖 原理高斯分布函数 极大似然估计求最优参数 朴素贝叶斯法 用同一个协方差三步骤 模型的简化 回归问题 大概内容与之前的类…

工业ESD防静电闸机系统帮助您实现静电监控自如!

如何有效检测、消除静电&#xff0c;对元器件品质进行有效管控&#xff1f;现在工厂常用的设备有腕带接地、电阻测试仪&#xff0c;ESD防静电系统&#xff0c;但是这些都是在我们生产的同时去监控处理掉静电&#xff0c;那么有没有一种能直接从人体根源消除掉静电的系统呢&…

Postman 的简单使用

什么是Postman 在程序开发中用于调试网络程序或者跟踪网页请求。可以对网页进行简单的基本信息调试。Postman最早是作用chrome浏览器插件存在的&#xff0c;但是2018年初Chrome停止对Chrome应用程序的支持。所以现在Postman提供了独立的安装包&#xff0c;不再依赖于Chrome浏览…

MySQL innodb引擎架构分析-Buffer Pool

文章目录系列文章目录前言一、Buffer Pool是什么&#xff1f;二、Buffer Pool是如何工作的&#xff1f;1. free链2. flush链根据脏页的情况(阈值)和相关配置刷新自适应刷脏3. Lru升级链总结参数&#xff1a;innodb_buffer_pool_size参数&#xff1a;innodb_buffer_pool_instanc…

避免重要数据泄露的8种方式

得益于国家的大力支持&#xff0c;我国数字化经济已开启高速发展模式&#xff0c;数据已经成为企业不可或缺的重要资产&#xff0c;相对应的数据安全风险问题也随之而来。 数据泄露不仅给企业带来了财产损失&#xff0c;也给企业带来了极大的声誉威胁。以下是日常生活中企业和…

W公司对接奥迪Audi EDI项目案例

项目背景 奥迪&#xff08;Audi&#xff09;是德国大众汽车集团子公司奥迪汽车公司旗下的豪华汽车品牌&#xff0c;作为高技术水平、质量标准、创新能力、以及经典车型款式的代表&#xff0c;奥迪是汽车品牌之一。公司总部设在德国的英戈尔施塔特&#xff0c;并在中国等许多国…

ASA防火墙高级配置——NAT控制欲NAT豁免

作者简介&#xff1a;一名在校云计算网络运维学生、每天分享网络运维的学习经验、和学习笔记。 座右铭&#xff1a;低头赶路&#xff0c;敬事如仪 个人主页&#xff1a;网络豆的主页​​​​​​ 目录 前言 一.NAT控制欲NAT豁免 1.禁用NAT控制 1&#xff09;出站(Outbou…

2021全国高校计算机能力挑战赛(初赛java)(部分)

数据1 input: 4 L G D U 7 L Y C N 2 T G E N 1 T R E P 4 output: 1 1 数据2 input: 6 L G D E 7 L Y C N 4 T G E N 4 T Y E P 3 L Y C N 2 T G D N 3 output: 1 1 package com.第四届校内模拟;import java.util.Scanner;/*** Author Lunau* Create 2022-12-09 18:02* Descri…

使用 ESP-DL 深度学习库基于 ESP32-S3 实现手势识别

人工智能改变了计算机与现实世界交互的方式。过去&#xff0c;人们通过微小的低功率设备和传感器获取数据&#xff0c;并传输至云端进行决策。这样的方式在设备连接性、成本和数据隐私方面带来了一定挑战。相对地&#xff0c;边缘人工智能是在物理设备上另一种处理数据的方式&a…

11.javase_面向对象1

一.面向对象介绍 1.1并不是一个技术&#xff0c;而是一种编程指导思想 1.2以什么形式组织代码&#xff1b;以什么思路解决问题 1.3为什么要用面向对象编程&#xff1a; 因为生活中&#xff0c;我们解决问题时&#xff0c;就是采用这种指导思想去解决的。 所以&#xff0c;我们写…

Codeforces Round #792 (Div. 1 + Div. 2)

比赛链接&#xff1a;Dashboard - Codeforces Round #792 (Div. 1 Div. 2) - Codeforces A: 思维 题意&#xff1a;Alice和Bob在玩游戏&#xff0c;每次等Alice交换两个不同位置的数后&#xff0c;Bob就会删除最后一个位置上的数&#xff0c;当最终只剩下一个数后&#xff0c…

基于对抗互信息学习特征解耦的零样本语音转换|INTERSPEECH 2022

零样本语音转换在转换时只需目标说话人的一段语音&#xff0c;更适合实际应用需求&#xff0c;具有广泛应用前景。 现有工作只考虑内容和音色表征的解耦&#xff0c;没有考虑与语音旋律相关的 韵律 和 音高 表征&#xff0c;导致与音高和韵律相关的信息泄露到音色中。 本文利用…

Flutter - Stack 与 Positioned 层叠布局

1 层叠布局和 Web 中的绝对定位、Android 中的 Frame 布局是相似的 2 子组件可以根据距父容器四个角的位置来确定自身的位置。 3 层叠布局允许子组件按照代码中声明的顺序堆叠起来。 4 Flutter中使用Stack和Positioned这两个组件来配合实现绝对定位。 5 Stack允许子组件堆叠&am…

什么是数据为先营销?为什么它对市场部如此重要?

数据为先营销希望能帮助各地的市场人员与高管们建立联系&#xff0c;实现这些商业目标&#xff0c;并真正学会如何适当地跟踪一切数据&#xff0c;这样他们就知道自己是否真的达到了收入目标&#xff0c;ROI&#xff0c;这些商业目标&#xff0c;而不是仅仅说我们是被数据驱动型…

先进的数字隔离技术提高了太阳能逆变器的可靠性

介绍 一个多世纪以来&#xff0c;化石燃料发电设施一直被证明是坚固可靠的能源&#xff0c;但这些久经考验的电力设施规模大、复杂&#xff0c;而且建造成本越来越高。以最小的碳足迹和环境影响清洁地操作它们也具有挑战性和成本高昂。相比之下&#xff0c;现代光伏&#xff0…

CSDN涨粉就这几招

目录说明涨粉不可不知的几件事几个常规的涨粉之道CSDN的数据运营之道粉丝列表关注列表关注某人取消关注获得某个用户的主要信息&#xff1a;昵称、排名、原创数、粉丝数等其它数据有了数据&#xff0c;怎么涨粉&#xff1f;说明 直到今年&#xff0c;我才开始重视涨粉&#xf…

_2LeetCode代码随想录算法训练营第二天C++

_2LeetCode代码随想录算法训练营第二天C LeetCode 题目列表&#xff1a; 977.有序数组的平方209.长度最小的子数组59.螺旋矩阵II 977.有序数组的平方 题目所述数组含有负数。 双指针的思路 双指针的思路&#xff1a; 最大元素一定是在两边&#xff0c;考虑用两个指针逐步…

PHY寄存器解读

以太网PHY寄存器分析 1 1、以太网PHY标准寄存器分析 2 1.1 Control Register 2 1.2 Status register 5 1.3 PHY Identifier Register 8 1.4 Auto-Negotiation Advertisement Register 8 1.5 Auto-Negotiation Link Partner Base Page Ability Register…

NY CREATE和Bleximo宣布达成新量子计算研发合作

&#xff08;图片来源&#xff1a;网络&#xff09; 12月7日&#xff0c;全栈量子计算系统集成公司Bleximo Corp.宣布&#xff1a;计划将其原型设计和营销业务扩展到纽约北部的奥尔巴尼纳米技术综合体(Albany NanoTech Complex)。该公司还将与纽约研究、经济发展、技术、工程和…

llvm编译、自带例子toy、llvm编译报错解决、.lib中搜索指定函数名

hunterzju/llvm-tutorial cs.cmu.edu/academic/class/15745-s14/public/lectures 编译llvm cd /d d:\llvm-home\ git clone gitgitcode.net:pubz/llvm-project.gitcd /d d:\llvm-home\llvm-project\ git status #HEAD detached at llvmorg-11.0.0set PATH%PATH%;D:\Python38\S…