React中控制子組件顯示隱藏的兩種方式及對比詳解
方式一:子組件觸發(fā)函數(shù)修改父組件狀態(tài)
先來看一段代碼:
// 父組件
import React, { useState } from 'react';
import Modal from './Modal';
const ParentComponent = () => {
const [isModalVisible, setIsModalVisible] = useState(false);
const handleOpenModal = () => {
setIsModalVisible(true);
};
const handleCloseModal = () => {
setIsModalVisible(false);
};
return (
<div>
<button onClick={handleOpenModal}>打開模態(tài)框</button>
{isModalVisible && <Modal onClose={handleCloseModal} />}
</div>
);
};
export default ParentComponent;
// 子組件
import React from 'react';
const Modal = ({ onClose }) => {
return (
<div className="modal">
<p>這是一個模態(tài)框</p>
<button onClick={onClose}>關閉模態(tài)框</button>
</div>
);
};
export default Modal;
在這段代碼中,父組件來維護狀態(tài) isModalVisible,當子組件通過調用onClose 時來設置 isModalVisible為true/false。從而實現(xiàn)子組件的顯示或者隱藏。
優(yōu)點:
每次重新銷毀組件重建直接通過isModalVisible && <Modal/> 控制組件掛載/卸載。對于需要每次展示都重置內部狀態(tài)(如表單)的彈窗,這種銷毀重建的方式更符合預期。
痛點:
每次子組件觸發(fā)onClose時,父組件的狀態(tài)(isModalVisible)都會變化,進而導致父組件及其所有子組件重新渲染。即使這個彈窗的顯隱邏輯完全獨立于父組件的其他邏輯,父組件仍然會被迫更新。
如果彈窗的顯隱邏輯完全屬于子組件自身(比如一個“確認刪除”彈窗,點擊按鈕后才觸發(fā)關閉),那么讓父組件管理這個狀態(tài)就顯得多余,增加了不必要的代碼復雜度。
方案二:子組件自治——forwardRef
另一種思路是讓子組件自管理狀態(tài),通過useImperativeHandle暴露控制方法給父組件,看一段代碼:
// 子組件
const Modal = forwardRef((props, ref) => {
const [visible, setVisible] = useState(false);
useImperativeHandle(ref, () => ({
open: () => setVisible(true),
close: () => setVisible(false)
}));
return visible ? (
<div className="modal">
<button onClick={() => setVisible(false)}>關閉</button>
</div>
) : null;
});
// 父組件
function Parent() {
const modalRef = useRef();
return (
<div>
<button onClick={() => modalRef.current?.open()}>打開彈窗</button>
<Modal ref={modalRef} />
</div>
);
}
優(yōu)點:
父組件極簡主義父組件無需維護任何狀態(tài),尤其適合多個彈窗的場景。調用modalRef.current.open() 簡單直接,避免狀態(tài)聲明污染,也可以避免父組件及其所有子組件重新渲染。
痛點:
- 打破組件封裝性父組件對子組件內部方法了如指掌,形成緊耦合。一旦子組件重構方法名,所有父組件都需要同步修改。
- 代碼復雜度每個組件都需要包裹forwardRef ,對于代碼開發(fā)不是特別方便。
如何抉擇?
父組件狀態(tài)管理:
- 彈窗顯隱與父組件狀態(tài)強相關(如表單提交成功后才展示)
- 需要嚴格遵循單向數(shù)據流,方便狀態(tài)追溯
- 彈窗內部需要每次打開重置狀態(tài)
forwardRef方案:
- 同一彈窗在多個分散位置觸發(fā)(如頁面頭部和底部都有觸發(fā)按鈕)
- 彈窗需要保持內部狀態(tài)(如填寫了一半的評論框臨時關閉)
- 父組件層級過深,prop drilling成本過高
總結
到底是使用父組件維護還是使用forwardRef, 這取決于具體項目業(yè)務的需要 , 如果你的項目里彈窗的顯隱邏輯更多是子組件自己的事,而不是父組件的核心邏輯,那么forwardRef可能是更優(yōu)雅的解決方案。反之,如果彈窗的開關直接影響父組件的核心狀態(tài)(如表單提交、數(shù)據加載),那么父組件管理仍然是更可靠的選擇。
以上就是React中控制子組件顯示隱藏的兩種方式及對比詳解的詳細內容,更多關于React控制子組件顯示隱藏的資料請關注腳本之家其它相關文章!
相關文章
React-router 4 按需加載的實現(xiàn)方式及原理詳解
本篇文章主要介紹了React-router 4 按需加載的實現(xiàn)方式及原理詳解,非常具有實用價值,需要的朋友可以參考下2017-05-05
解決React報錯`value` prop on `input` should&
這篇文章主要為大家介紹了React報錯`value` prop on `input` should not be null解決方法詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2022-12-12

