【React】valtio快速上手
admin
2024-02-22 08:29:38
0

前言

  • 现在有很多人抛弃redux转向valtio,包括Umi最新版也开始使用它。
  • react状态管理门派一般分为以下几类:
    没有状态管理工具:直接用 props 或者 context
    单项数据流:redux、zustand
    双向绑定:mobx、valtio
    状态原子化:jotai、recoil
    有限状态机:xstate
  • 我觉得一个好的状态管理器要有超低的学习成本、能产生符合预期的效果、并且性能不会很差。
  • valtio和jotai 是同一个作者,今天主角valtio是以proxy为核心的状态管理库。
  • valtio由于以proxy为核心,所以可以脱离react使用。
  • 这里摘抄云谦大佬对umi加入valtio的原话:
    1、数据与视图分离的心智模型。不再需要在 React 组件或 hooks 里用 useState 和 useReducer 定义数据,或者在 useEffect 里发送初始化请求,或者考虑用 context 还是 props 传递数据。如果熟悉 dva,你会在此方案中找到一丝熟悉的感觉,概念上就是 reducer 和 effects 换成了 actions,subscriptions 则换了种形式存在。
    2、更现代的 dva?「现代」主要出于这些点,1)基于 proxy,mutable,所以更新更简单,同时读取更高效,无需 selector,tracking with usage,2)没有中心化的 actions,以及基于组合式的扩展机制,这些都是对 TypeScript 更友好的方式,3)更少脚手架代码。
    再看缺点。
    1、兼容性。由于基于 proxy,所以不支持 IE 11、Chrome 48、Safari 9.1、Firefox 17 和 Opera 35 等。如果你的项目目前或未来有兼容需求,不要用。
    2、非 Hooks 数据流。这不一定算缺点,看从什么角度看。但如果你是从 useModel、hox 这类 Hooks 数据流切过来,会发现有些事情不能做了。不能在 state 里组合其他的 hooks 数据,也不能在 actions 调用其他的 hooks 方法。我就想用 hooks 的方式组织,怎么办?解法是把 valtio 的 store 作为一个原子,和其他 hook 结合使用,而不是在 store 里调用其他 hook。

使用

安装

npm i valtio

proxy与useSnapshot

  • 使用proxy包装需要代理的对象。
  • 在任意的地方去进行更改。
  • 在需要刷新显示的地方使用useSnapshot监听该对象变化
import React, { useEffect, useState } from "react";
import { proxy, useSnapshot } from "valtio";const state = proxy({ count: 0, text: "hello" });setInterval(() => {++state.count;
}, 1000);export default function App() {const snap = useSnapshot(state);console.log("refresh");return (
{ height: 500, padding: 20 }}>{snap.count}
); }
  • 可以发现深度的检测也是可以的:
import React, { useEffect, useState } from "react";
import { proxy, useSnapshot } from "valtio";const state = proxy({ count: 0, text: "hello", aaa: { bbb: 1 } });
setInterval(() => {state.aaa.bbb++;
}, 1000);function Child1() {console.log("child1 refresh");return 
child1
; } function Child2() {console.log("child2 refresh");return
child2
; } export default function App() {const snap = useSnapshot(state);console.log("refresh");return (
{ height: 500, padding: 20 }}>{snap.aaa.bbb}
); }
  • 其余行为符合react规律,如果不依赖变化的属性,该组件也不会刷新。

subscribe与subscribeKey

  • 订阅顾名思义,任何地方使用后改变其然后执行函数。
import React, { useEffect, useState } from "react";
import { proxy, useSnapshot, subscribe } from "valtio";const state = proxy({ count: 0, text: "hello", aaa: { bbb: 1 } });
setInterval(() => {state.aaa.bbb++;
}, 1000);function Child1() {console.log("child1 refresh");return 
child1
; } function Child2() {console.log("child2 refresh");return
child2
; } export default function App() {console.log("refresh");useEffect(() => {const unsubscribe = subscribe(state, () => {console.log("jjjjjjjjj", state.aaa.bbb);});return () => unsubscribe();}, []);return (
{ height: 500, padding: 20 }}>
); }
  • 调试发现第一个只能接对象。如果state内部的对象没更新他也可以不更新。
  • 如果想要subscribe对象外的可以使用subscribeKey解决。
import React, { useEffect, useState } from "react";
import { proxy, useSnapshot, subscribe } from "valtio";
import { subscribeKey } from "valtio/utils";
const state = proxy({count: 0,text: "hello",aaa: { bbb: 1 },ccc: { dd: 1 },
});
setInterval(() => {state.count++;
}, 1000);function Child1() {console.log("child1 refresh");return 
child1
; } function Child2() {console.log("child2 refresh");return
child2
; } export default function App() {console.log("refresh");useEffect(() => {const unsubscribe = subscribeKey(state, "count", (v) => {console.log("jjjjjjjjj", v);});return () => unsubscribe();}, []);return (
{ height: 500, padding: 20 }}>
); }

watch

  • watch可以拿到索要的state,只要它进行了变化:
import React, { useEffect, useState } from "react";
import { proxy, useSnapshot, subscribe } from "valtio";
import { subscribeKey, watch } from "valtio/utils";
const state = proxy({count: 0,text: "hello",aaa: { bbb: 1 },ccc: { dd: 1 },
});
const state2 = proxy({count: 0,text: "hello",
});
setInterval(() => {state.count++;
}, 1000);function Child1() {console.log("child1 refresh");return 
child1
; } function Child2() {console.log("child2 refresh");return
child2
; } export default function App() {console.log("refresh");useEffect(() => {const stop = watch((get) => {console.log("state has changed to", get(state2),); // auto-subscribe on use});return () => stop();}, []);return (
{ height: 500, padding: 20 }}>
);

ref

  • 如果这个东西你不想被proxy代理又想取值,那么可以使用ref进行包裹:
import { proxy, ref } from 'valtio'const state = proxy({count: 0,dom: ref(document.body),
})

取消批量更新

  • 默认情况是开启的,但是如果是输入框这种情况,当你键入一些值后,光标移动到之前地方输入,光标还会跳转到最末尾。使用sync则可以解决这个问题。
function TextBox() {const snap = useSnapshot(state, { sync: true });console.log("mmmm");return (snap.text}onChange={(e) => (state.text = e.target.value)}/>);
}

derive

  • 派生类似于computed可以从一个被proxy的值里进行变化。
  • 配合snapshot可以进行react组件刷新
import React, { useEffect, useState } from "react";
import { proxy, useSnapshot, subscribe } from "valtio";
import { subscribeKey, watch, derive } from "valtio/utils";
// create a base proxy
const state = proxy({count: 1,
});// create a derived proxy
const derived = derive({doubled: (get) => get(state).count * 2,
});// alternatively, attach derived properties to an existing proxy
const three = derive({tripled: (get) => get(state).count * 3,},{proxy: state,}
);setInterval(() => {++state.count;
}, 1000);export default function App() {const snap = useSnapshot(state);return (
{ height: 500, padding: 20 }}>{derived.doubled}============{three.tripled}=============={three.count}{snap.count}
); }

proxyWithComputed

  • 类似于computed,与上面相比多了缓存。
  • 作者建议尽量不要使用,因为和usesnapshot做了相同的事,而且有可能会导致内存泄漏
import memoize from 'proxy-memoize'
import { proxyWithComputed } from 'valtio/utils'const state = proxyWithComputed({count: 1,},{doubled: memoize((snap) => snap.count * 2),}
)// Computed values accept custom setters too:
const state2 = proxyWithComputed({firstName: 'Alec',lastName: 'Baldwin',},{fullName: {get: memoize((snap) => snap.firstName + ' ' + snap.lastName),set: (state, newValue) => {;[state.firstName, state.lastName] = newValue.split(' ')},},}
)// if you want a computed value to derive from another computed, you must declare the dependency first:
const state = proxyWithComputed({count: 1,},{doubled: memoize((snap) => snap.count * 2),quadrupled: memoize((snap) => snap.doubled * 2),}
)

proxyWithHistory

  • proxyWithHistory 自带提供了redo和undo命令。
import { proxyWithHistory } from 'valtio/utils'const state = proxyWithHistory({ count: 0 })
console.log(state.value) // ---> { count: 0 }
state.value.count += 1
console.log(state.value) // ---> { count: 1 }
state.undo()
console.log(state.value) // ---> { count: 0 }
state.redo()
console.log(state.value) // ---> { count: 1 }

proxySet与proxyMap

  • 这2都是提供创建proxy对象能力
import { proxySet } from 'valtio/utils'const state = proxySet([1, 2, 3])
//can be used inside a proxy as well
//const state = proxy({
//    count: 1,
//    set: proxySet()
//})state.add(4)
state.delete(1)
state.forEach((v) => console.log(v)) // 2,3,4
import { proxyMap } from 'valtio/utils'const state = proxyMap([['key', 'value'],['key2', 'value2'],
])
state.set('key', 'value')
state.delete('key')
state.get('key') // ---> value
state.forEach((value, key) => console.log(key, value)) // ---> "key", "value", "key2", "value2"

总结

  • valtio提供了proxy封装的各种用法,使用上较为简单,也容易满足需求。对于状态库要求不是特别高,以及对浏览器支持要求不高时,该状态库比较适合使用。

相关内容

热门资讯

路透解析“马斯克集团”:Spa... SpaceX 凤凰网科技讯 北京时间1月31日,据路透社报道,长期以来,埃隆·马斯克(Elon Mu...
启动“二改” 永辉在京完成21... 北京商报讯(记者 赵述评 实习记者 毛思怡)1月31日,永辉超市北京龙湖长楹天街店经一个多月闭店调改...
《宜宾散装白酒连锁经营规范》团... 近日,由宜宾市酒类协会牵头归口、宜宾安宁酒厂主导起草,四川谊宾酒业、宜宾学院、劲牌南溪酒业等多家本地...
印度牙医博士打造全印首款人形机... 2026 年 1 月 23 日,印度浦那的 Muks Robotics 正式宣布,自主研发的社交人形...
金银价创新高,引发全球“贵金属... 【环球时报记者 倪浩 环球时报特约记者 甄翔】连日来,国际市场金银价格持续大涨。1月29日当天,亚太...
财经观察丨“爱你老己”背后的消... 新华网北京1月31日电岁末年初,一句“爱你老己,明天见”席卷社交网络,成为年轻人自我关怀的新表达。热...
重磅!珠海科技产业集团与农行广... 1月30日,珠海科技产业集团与中国农业银行广东省分行在广州签署全面战略合作协议暨独立授信合作。农行广...
原创 黄... 谁能想到,2026年开年就上演金融魔幻现实主义! 国际黄金1月31日凌晨暴跌9.25%,盘中狂泻12...
云南省本级社会保险基金银行存款... 近日,云南省财政厅、云南省人力资源和社会保障厅、云南省医疗保障局联合印发《云南省本级社会保险基金银行...
病毒在身体里“安家”却相安无事... 很多人听说“乙肝携带者”,总会下意识和“乙肝患者”画上等号,担心自己或身边人被传染,也害怕携带者最终...
库迪确认:取消全场9.9元 来源:滚动播报 (来源:新消费日报) 有消息称,库迪咖啡发布门店价格策略和活动调整通知。通知指出,...
原创 雷... 不知道大家有没有发现,这个周六可能是进入2026年之后最消停的一个周六。因为各品牌基本上都没什么大事...
原创 特... 特朗普对委内瑞拉的举动,表面上看是一场能源棋局,实则背后隐藏着深刻的战略考量。对他而言,掌握能源就意...
原创 李... 01、“私募魔女”李蓓再引争议 半夏投资创始人、“私募魔女”李蓓,最近又成为投资圈的焦点。 1月2...
爱美客:AestheFill产... 上证报中国证券网讯(记者 王子霖)备受医美行业瞩目的AestheFill产品独家经销权纠纷迎来重要进...
雷军明晚直播,在北京小米汽车工... IT之家 1 月 31 日消息,今天午间,小米创办人、董事长兼 CEO 雷军在微博发文宣布,2 月 ...
字节阿里DeepSeek决战春... 新智元报道 编辑:艾伦 【新智元导读】这个春节,中国 AI 迎来「决战时刻」。据《The Info...
皇台酒业开始过年? 富凯摘要:有钱没钱喝酒过年。 作者|欧文 1月30日,白酒板块再现分化行情,皇台酒业却延续强势表现,...
深交所修订可持续发展报告编制指... 上证报中国证券网讯 据深交所1月30日消息,深交所发布实施《深圳证券交易所上市公司自律监管指南第3号...
面试餐饮|新手零经验,小红书开... 有没有餐饮人跟我一样?想靠小红书引流拓客,却卡在第一步:不知道怎么开店、怎么发笔记不踩雷,看着别人的...