【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封装的各种用法,使用上较为简单,也容易满足需求。对于状态库要求不是特别高,以及对浏览器支持要求不高时,该状态库比较适合使用。

相关内容

热门资讯

测血糖怕疼、嫌麻烦?这6个方法... “一天扎好几次手指,手指头都快成筛子了……” “每次测血糖之前都紧张半天,下不去手。” “上班忙、带...
洋河迎来价值重估时刻:从消费者... 5月12日,洋河股份举行2025年度网上业绩说明会。洋河股份董事长、总裁顾宇带领管理层集体出席。 这...
四大指数集体收跌,沪指失守42... 每经记者:杜波 记者|杜波 编辑|陈柯名杜恒峰校对|程鹏 14日,A股市场震荡调整,截至收盘,创业...
原创 5... 写在文章前的声明:在本文之前的说明:本文中所列的投资信息,只是一个对基金资产净值进行排行的客观描述,...
信立泰“心衰神药”遇挫,“药茅... 千亿梦想还在。 作者 | 孙梦圆 编辑丨于婞 来源 | 野马财经 因为“心衰神药”临床不及预期,“慢...
阿里终于看到回头钱了 出品|虎嗅商业消费组 作者|苗正卿 题图|视觉中国 阿里正在获得一批新的“小弟牌印钞机”。 多年前投...
伊利、蒙牛盯上新赛道!盒马等已... 本文自南都·湾财社。 采写 | 南都·湾财社记者 王静娟 中国水饮市场又一细分赛道被巨头盯上。 近日...
伯希和更名,三战港股ipo,“... 出品丨搜狐财经 作者丨柴鑫洋 编辑丨李文贤 被称作“始祖鸟平替”的户外品牌“伯希和”,它的母公司“奔...
视频直播丨维通利5月15日深交... 来源:全景财经 维通利新股发行上市仪式 维通利新股发行上市仪式将于2026年5月15日08:57-0...
余额宝破“1”!1万1天仅挣2... 闲钱理财收益再创新低! Wind数据显示,自5月5日天弘余额宝7日年化收益率跌破1%后,该货币基金的...
航运股逆市拉升,发电设备等跌幅... 【导读】上午A股三大指数震荡回调,航运股逆市拉升,工业气体、猪肉概念股上涨 中国基金报记者 张舟 5...
原创 1... 去年,微软有整整50万块AI芯片,就这么趴在仓库里睡了18个月。不是买不到,是没电用。与此同时,中国...
ETF收评:沪指失守4200点... 5月14日,A股市场整体震荡走弱,创业板指、深成指跌幅均超2%,沪指失守4200点关口。截至当日收盘...
一季度我国工程机械产品出口额同... 今年一季度,我国工程机械行业国际化步伐持续加快,海外业务比重显著提升,工程机械产品出口保持快速增长。...
原创 1... 合肥只排在第11名,杭州连重庆都不敌?这组新一线城市排名让人感到惊讶,因为这几个城市个个都是当地的明...
微软被曝考虑收购大模型创企,S... 智东西 编译 | 佳扬 编辑 | 云鹏 智东西5月14日消息,据路透社今日报道,微软正加速布局AI初...
开局就面临“地狱级”大考!沃什... 面对难啃的通胀,华尔街对沃什深表忧虑,投资者正摩拳擦掌,准备迎接美国国债收益率将更高的新常态。 中东...
马斯克中文发帖:我的儿子正在学... 5月14日上午,特斯拉首席执行官马斯克携幼子现身人民大会堂,孩子身着新中式上衣。 当天稍晚,马斯克在...
“童鞋界爱马仕”冒犯打工人?泰... 在各大交通枢纽铺满广告的童鞋品牌泰兰尼斯,却因一则广告引发了争议。 近日,泰兰尼斯一则“送礼就送稳跑...