react usecontext_Vue3原理實戰運用,我用40行代碼把他裝進了React做狀態管理

前言

vue-next是Vue3的源碼倉庫,Vue3采用lerna做package的劃分,而響應式能力@vue/reactivity被劃分到了單獨的一個package中。

如果我們想把它集成到React中,可行嗎?來試一試吧。

使用示例

話不多說,先看看怎么用的解解饞吧。

//?store.ts
import?{?reactive,?computed,?effect?}?from?'@vue/reactivity';

export?const?state?=?reactive({
??count:?0,
});

const?plusOne?=?computed(()?=>?state.count?+?1);

effect(()?=>?{
??console.log('plusOne?changed:?',?plusOne);
});

const?add?=?()?=>?(state.count?+=?1);

export?const?mutations?=?{
??//?mutation
??add,
};

export?const?store?=?{
??state,
??computed:?{
????plusOne,
??},
};

export?type?Store?=?typeof?store;
//?Index.tsx
import?{?Provider,?useStore?}?from?'rxv'
import?{?mutations,?store,?Store?}?from?'./store.ts'
function?Count()?{
??const?countState?=?useStore((store:?Store)?=>?{
????const?{?state,?computed?}?=?store;
????const?{?count?}?=?state;
????const?{?plusOne?}?=?computed;

????return?{
??????count,
??????plusOne,
????};
??});

??return?(
????<Card?hoverable?style={{?marginBottom:?24?}}><h1>計數器h1><div?className="chunk"><div?className="chunk">store中的count現在是?{countState.count}div><div?className="chunk">computed值中的plusOne現在是?{countState.plusOne.value}div><Button?onClick={mutations.add}>addButton>div>Card>
??);
}

export?default?()?=>?{
??return?(
????<Provider?value={store}><Count?/>Provider>
??);
};

可以看出,store的定義只用到了@vue/reactivity,而rxv只是在組件中做了一層橋接,連通了Vue3和React,然后我們就可以盡情的使用Vue3的響應式能力啦。

預覽

844be8a7f14c824a68f00d7b6b463f7e.gif
gif

可以看到,完美的利用了reactive、computed的強大能力。

分析

從這個包提供的幾個核心api來分析:

effect(重點)

effect其實是響應式庫中一個通用的概念:觀察函數,就像Vue2中的Watcher,mobx中的autorunobserver一樣,它的作用是收集依賴

它接受的是一個函數,它會幫你執行這個函數,并且開啟依賴收集,

這個函數內部對于響應式數據的訪問都可以收集依賴,那么在響應式數據被修改后,就會觸發更新。

最簡單的用法

const?data?=?reactive({?count:?0?})
effect(()?=>?{
????//?就是這句話?訪問了data.count
????//?從而收集到了依賴
????console.log(data.count)
})

data.count?=?1
//?控制臺打印出1

那么如果把這個簡單例子中的

()?=>?{
????//?就是這句話?訪問了data.count
????//?從而收集到了依賴
????console.log(data.count)
}

這個函數,替換成React的組件渲染,是不是就能達成響應式更新組件的目的了?

reactive(重點)

響應式數據的核心api,這個api返回的是一個proxy,對上面所有屬性的訪問都會被劫持,從而在get的時候收集依賴(也就是正在運行的effect),在set的時候觸發更新。

ref

對于簡單數據類型比如number,我們不可能像這樣去做:

let?data?=?reactive(2)
//??oops
data?=?5

這是不符合響應式的攔截規則的,沒有辦法能攔截到data本身的改變,只能攔截到data身上的屬性的改變,所以有了ref。

const?data?=?ref(2)
//??ok
data.value=?5

computed

計算屬性,依賴值更新以后,它的值也會隨之自動更新。其實computed內部也是一個effect。

擁有在computed中觀察另一個computed數據、effect觀察computed改變之類的高級特性。

實現

從這幾個核心api來看,只要effect能接入到React系統中,那么其他的api都沒什么問題,因為它們只是去收集effect的依賴,去通知effect觸發更新。

effect接受的是一個函數,而且effect還支持通過傳入schedule參數來自定義依賴更新的時候需要觸發什么函數,如果我們把這個schedule替換成對應組件的更新呢?要知道在hook的世界中,實現當前組件強制更新可是很簡單的:

useForceUpdate

export?const?useForceUpdate?=?()?=>?{
??const?[,?forceUpdate]?=?useReducer(s?=>?s?+?1,?0);
??return?forceUpdate;
};

這是一個很經典的自定義hook,通過不斷的把狀態+1來強行讓組件渲染。

rxv的核心api: useStore接受的也是一個函數selector,它會讓用戶自己選擇在組件中需要訪問的數據。

那么思路就顯而易見了:

  1. selector包裝在effect中執行,去收集依賴。
  2. 指定依賴發生更新時,需要調用的函數是當前正在使用useStore的這個組件的forceUpdate強制渲染函數。

這樣不就實現了數據變化,組件自動更新嗎?

簡單的看一下核心實現

useStore和Provider

import?React,?{?useContext?}?from?'react';
import?{?useForceUpdate,?useEffection?}?from?'./share';

type?Selector?=?(store:?T)?=>?S;const?StoreContext?=?React.createContext(null);const?useStoreContext?=?()?=>?{const?contextValue?=?useContext(StoreContext);if?(!contextValue)?{throw?new?Error('could?not?find?store?context?value;?please?ensure?the?component?is?wrapped?in?a?',
????);
??}return?contextValue;
};/**
?*?在組件中讀取全局狀態
?*?需要通過傳入的函數收集依賴
?*/export?const?useStore?=?(selector:?Selector):?S?=>?{
??const?forceUpdate?=?useForceUpdate();
??const?store?=?useStoreContext();
??const?effection?=?useEffection(()?=>?selector(store),?{
????scheduler:?forceUpdate,
????lazy:?true,
??});
??const?value?=?effection();
??return?value;
};
export?const?Provider?=?StoreContext.Provider;

這個option是傳遞給Vue3的effectapi,

scheduler規定響應式數據更新以后應該做什么操作,這里我們使用forceUpdate去讓組件重新渲染。

lazy表示延遲執行,后面我們手動調用effection來執行


{
??scheduler:?forceUpdate,
??lazy:?true,
}

再來看下useEffectionuseForceUpdate

import?{?useEffect,?useReducer,?useRef?}?from?'react';
import?{?effect,?stop,?ReactiveEffect?}?from?'@vue/reactivity';

export?const?useEffection?=?(...effectArgs:?Parameters<typeof?effect>)?=>?{
??//?用一個ref存儲effection
??//?effect函數只需要初始化執行一遍
??const?effectionRef?=?useRef();if?(!effectionRef.current)?{
????effectionRef.current?=?effect(...effectArgs);
??}//?卸載組件后取消effectconst?stopEffect?=?()?=>?{
????stop(effectionRef.current!);
??};
??useEffect(()?=>?stopEffect,?[]);return?effectionRef.current
};export?const?useForceUpdate?=?()?=>?{const?[,?forceUpdate]?=?useReducer(s?=>?s?+?1,?0);return?forceUpdate;
};

也很簡單,就是把傳入的函數交給effect,并且在組件銷毀的時候停止effect而已。

流程

  1. 先通過useForceUpdate在當前組件中注冊一個強制更新的函數。
  2. 通過useContext讀取用戶從Provider中傳入的store。
  3. 再通過Vue的effect去幫我們執行selector(store),并且指定scheduler為forceUpdate,這樣就完成了依賴收集。
  4. 那么在store里的值更新了以后,觸發了scheduler也就是forceUpdate,我們的React組件就自動更新啦。

就簡單的幾行代碼,就實現了在React中使用@vue/reactivity中的所有能力。

優點:

  1. 直接引入@vue/reacivity,完全使用Vue3的reactivity能力,擁有computed, effect等各種能力,并且對于Set和Map也提供了響應式的能力。后續也會隨著這個庫的更新變得更加完善的和強大。
  2. vue-next倉庫內部完整的測試用例。
  3. 完善的TypeScript類型支持。
  4. 完全復用@vue/reacivity實現超強的全局狀態管理能力。
  5. 狀態管理中組件級別的精確更新。
  6. Vue3總是要學的嘛,提前學習防止失業!

缺點:

  1. 由于需要精確的收集依賴全靠useStore,所以selector函數一定要精確的訪問到你關心的數據。甚至如果你需要觸發數組內部某個值的更新,那你在useStore中就不能只返回這個數組本身。

舉一個例子:

function?Logger()?{
??const?logs?=?useStore((store:?Store)?=>?{
????return?store.state.logs.map((log,?idx)?=>?(
??????<p?className="log"?key={idx}>
????????{log}p>
????));
??});

??return?(
????<Card?hoverable><h1>控制臺h1><div?className="logs">{logs}div>Card>
??);
}

這段代碼直接在useStore中返回了整段jsx,是因為map的過程中回去訪問數組的每一項來收集依賴,只有這樣才能達到響應式的目的。

源碼地址

https://github.com/sl1673495/react-composition-api

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/370691.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/370691.shtml
英文地址,請注明出處:http://en.pswp.cn/news/370691.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

Spring MVC –自定義RequestMappingHandlerMapping

在xml bean定義文件中使用<mvc&#xff1a;annotation-driven />配置Spring MVC時&#xff0c;在內部將一個名為RequestMappingHandlerMapping的組件注冊到Spring MVC。 該組件或通常是HandlerMapping組件負責將請求URI路由到處理程序&#xff0c;這些處理程序是使用Requ…

css的三個特性 背景透明設置

關于行內元素&#xff08;補充一點&#xff09; 行內元素只能容納文本或其他行內元素。&#xff08;a特殊a里面可以放塊級元素&#xff09; 例子&#xff1a; 關于行高tip: 選擇器的嵌套層級不應大于3級&#xff0c;位置靠后的限定條件應盡可能的精確。 屬性定義必須另起一行…

比較容易犯的一些智障錯誤(不定時修改)

無論在什么學習中&#xff0c;在成長的過程中&#xff0c;注定要犯一些錯誤&#xff0c;有些比較高級的錯誤&#xff0c;有些是比較智障的錯誤。那么在oi的學習中&#xff0c;我們最討厭的就是一些智障的小錯誤&#xff0c;因為如果是大錯誤的話一般情況下在測試樣例的時候都是…

ccs安裝多版本編譯器離線_大數據分析:學習工具JDK,在線安裝指南

hadoop是使用Java語言開發的并且Hadoop運行需要有Java環境的支持&#xff0c;因此在安裝hadoop之前需要安裝Java開發環境即JDK(Java Development Kit)。安裝前首先向大家介紹以一下本文會用到的幾個詞&#xff1a;JAVA_HOME:一是為了方便引用&#xff0c;比如&#xff0c;JDK安…

HTML基礎入門學習準備篇

在學習前端的開始&#xff0c;讓我們一起來了解什么是HTML5時代的大前端開發和全棧開發的定義傳統的前端&#xff1a;切圖-標簽和樣式-實現效果H5時代的前端&#xff1a;一、需要各端的兼容開發二、可以用于APP開發和移動站點的開發三、Ajax服務器端技術開發四、高級設計模式和…

asp.net尚未在web服務器上注冊_最新版Web服務器項目詳解 00 項目概述

點 擊 關 注 上 方&#xff02;兩猿社&#xff02;設 為&#xff02;置 頂 或 星 標&#xff02;&#xff0c;干 貨 第 一 時 間 送 達。互 聯 網 猿 | 兩 猿 社TineyWebServerLinux下C輕量級Web服務器&#xff0c;助力初學者快速實踐網絡編程&#xff0c;搭建屬于自己的服務器…

python正則r的作用_Python正則表達式,這一篇就夠了!

原標題&#xff1a;Python正則表達式&#xff0c;這一篇就夠了&#xff01;大多數編程語言的正則表達式設計都師從Perl&#xff0c;所以語法基本相似&#xff0c;不同的是每種語言都有自己的函數去支持正則&#xff0c;今天我們就來學習 Python中關于 正則表達式的函數。re模塊…

服務器微信了早上好,每天早上好的問候語 微信早安問候語合集66句

1、沒有傘的孩子&#xff0c;必須努力奔跑&#xff01;早安&#xff01;2、你不能改變過去&#xff0c;但你可以改變未來。早安&#xff01;3、堅持了才叫夢想&#xff0c;放棄了就只是妄想。早安&#xff01;4、忘掉失敗&#xff0c;不過要牢記失敗中的教訓。早安&#xff01;…

如何得到某個文件的舊版本

下載某個文件的舊版本 如果想要得到某個文件的舊版本&#xff0c;只需在該文件上單擊右鍵&#xff0c;選擇Updata to revision…即可。 系統會提示輸入版本號。 例如要下載soc_1的第五個版本&#xff0c;只需填入5即可。如圖7。 查看完版本5的文件后&#xff0c;如果想在此回到…

Google Guava EventBus和Java 7 WatchService用于事件編程

這篇文章將介紹如何使用Guava EventBus將更改發布到Java 7 WatchService檢測到的目錄或子目錄中。 Guava EventBus是向應用程序添加發布/訂閱通信的好方法。 Java 7 java.nio.file軟件包中新增的WatchService用于監視目錄中的更改。 由于EventBus和WatchService已在以前的文章中…

Bootstrap-table 部分瀏覽器顯示不出來

一、問題 近日&#xff0c;寫了一個ASP.Net項目&#xff0c;但是bootstrap-table在別人的電腦上顯示不出來&#xff0c;在自己的電腦上能顯示&#xff0c;有些瀏覽器也是能顯示&#xff0c;但部分瀏覽器就是顯示不出來。找了很多原因&#xff0c;最后有個老師和我說是內核版本的…

DBMS-基本概念

文件處理系統&#xff08;file-processing system&#xff09;的主要弊端&#xff1a; 數據冗余和不一致&#xff08;data redundancy and inconsistency&#xff09;、數據訪問困難&#xff08;difficulty in accessing data&#xff09;、數據孤立&#xff08;data isolation…

python多大孩子可以學_孩子學編程最佳年齡是多大

兒童編程教育已經悄悄地掀起了編程低齡化的熱浪。但是很多人會疑惑&#xff0c;到底孩子學編程最佳年齡是多少呢&#xff1f;下面小編就為大家解答一下。孩子學編程最佳年齡 首先&#xff0c;孩子的學習黃金時期是非常重要的。從6歲開始&#xff0c;大多數孩子都可以掌握對基本…

上傳文件到華為云云服務器,上傳文件到云服務器

上傳文件到云服務器 內容精選換一換登錄Windows操作系統的彈性云服務器時&#xff0c;需使用密碼方式登錄。因此&#xff0c;用戶需先根據創建彈性云服務器時使用的密鑰文件&#xff0c;獲取該彈性云服務器初始安裝時系統生成的管理員密碼(Administrator帳戶或Cloudbase-init設…

JUnit規則

第一次偶然發現JUnit Rule批注時&#xff0c;我對此概念有些惱火。 在測試用例中擁有一個公共領域似乎有些奇怪&#xff0c;因此我不愿意定期使用它。 但是一段時間后&#xff0c;我習慣了這一點&#xff0c;事實證明&#xff0c;規則可以通過多種方式簡化編寫測試的過程。 這篇…

微pe工具箱是微軟的嗎_微PE工具箱V2.0更新10內核

本帖最后由 韋小寶2 于 2017-3-22 21:26 編輯微PE工具箱V2.0發布更新了&#xff0c;鏈接&#xff1a;http://pan.baidu.com/s/1c9whpO 轉載自無憂論壇 07年&#xff0c;第一個XP內核通用PE工具箱發布。17年&#xff0c;第一個WIN10內核的微PE工具箱就此誕生。64位純內核&#x…

jq實現輪播圖

之前設計了一個校團委網站&#xff0c;里面有一個輪播圖效果&#xff0c;上網后查看了許多方法&#xff0c;覺得下面這個方法最為適用&#xff0c;記錄下來 js代碼&#xff1a; $(document).ready(function () {//無縫切換輪播var i 0;//索引var clone $(".banner .imgL…

關于圖片預加載的思考

引子&#xff1a; 很多時候&#xff0c;我們在寫html頁面的時候&#xff0c;當需要在頁面中加入圖片時&#xff0c;我們很自然地把圖片直接用<img>標簽放在了<body>里面&#xff0c;這本來是沒有多大問題的。 但是當圖片數量很多的時候&#xff0c;問題就來了。H…

oracle安裝中桌面模式與服務器模式的去別

桌面模式只能本機使用。 服務器模式可以在網絡中使用&#xff0c;也就是網絡中的其他服務器可以使用。 所以安裝時&#xff0c;如果是生產環境肯定是服務器模式。一般也都是服務器模式。 轉載于:https://www.cnblogs.com/zhjx0521/p/7803691.html

python不能創建字典的是_用Python創建帶有重復鍵的字典

用Python創建帶有重復鍵的字典 我有以下列表&#xff0c;其中包含重復的具有不同值的汽車注冊號。 我想將其轉換為字典&#xff0c;該字典接受汽車登記號的多個鍵。 到目前為止&#xff0c;當我嘗試將列表轉換為字典時&#xff0c;它消除了鍵之一。 如何制作具有重復鍵的字典&a…