- 資訊首頁(yè) > 開(kāi)發(fā)技術(shù) > web開(kāi)發(fā) > JavaScript >
- React中Portals與錯誤邊界處理實(shí)現
可以說(shuō)是 插槽,但 不同于 Vue 中的 slot,它指的是將一個(gè) React 元素渲染到指定的容器 (真實(shí) DOM) 中
比如說(shuō),Modal 組件一般默認直接作為 body 的真實(shí)結構的子元素渲染出來(lái),那么我們就可以借助 ReactDOM.createPortal(ReactElement, RealDOM container) 創(chuàng )建一個(gè) React 元素,示例代碼:
import React from 'react' import ReactDOM from 'react-dom' import Modal from './components/Modal' const PortalModal = ReactDOM.createPortal(<Modal />, document.body) export default function App() { return <div className="app-container"> <PortalModal /> </div> }
我們可以在瀏覽器控制臺中看到,真實(shí)的 Modal 組件其實(shí)是作為 body 的直接子元素渲染出來(lái)的,但通過(guò) React 開(kāi)發(fā)者工具,我們可以看到 Modal 組件在虛擬 DOM 樹(shù)的結構中依舊在 App 組件下,類(lèi)名為 app-container 的 div 中
所以,我們可以得出結論:React 組件虛擬 DOM 樹(shù)結構與真實(shí) DOM 樹(shù)結構可以是不一致的
因而需要注意事件冒泡
默認情況下,若一個(gè)組件在渲染期間 (render) 發(fā)生錯誤,那么就會(huì )導致整個(gè)組件樹(shù)全部被卸載
錯誤邊界:就是一個(gè)組件,用于捕獲 渲染期間 子組件發(fā)生的錯誤,并有能力阻止錯誤繼續向父組件傳播
讓某個(gè)組件捕獲錯誤 (類(lèi)組件):
使用靜態(tài)方法 static getDerivedStateFromError,子組件渲染錯誤時(shí)會(huì )觸發(fā)此函數
import React, {PureComponent} from 'react' export default class ErrorBoundary extends PureComponent { state = { isError: false } static getDerivedStateFromError(error) { console.log('Rendering Error: ', error) return { isError: true } } render() { if (this.isError) { return <span>Something Wrong...</span> } return this.props.children } }
使用 componentDidCatch(error, info) 函數
import React, {PureComponent} from 'react' export default class ErrorBoundary extends PureComponent { state = { isError: false } componentDidCatch(error, info) { // info 即為錯誤的摘要信息 console.log('Rendering Error: ', error) console.log('Rendering info: ', info) this.setState({ isError: true }) } render() { if (this.isError) { return <span>Something Wrong...</span> } return this.props.children } }
自 React 16 起,任何未被錯誤邊界捕獲的錯誤將會(huì )導致整個(gè) React 組件樹(shù)被卸載。
經(jīng)驗告訴我們,完全移除比保留錯誤UI更好。例如,在類(lèi)似 Messenger 的產(chǎn)品中,把異常的 UI 展示給用戶(hù)可能會(huì )導致用戶(hù)將信息錯發(fā)給別人。
增加錯誤邊界能夠讓你在應用發(fā)生異常時(shí)提供更好的用戶(hù)體驗。例如,Facebook Messenger 將側邊欄、信息面板、聊天記錄以及信息輸入框包裝在單獨的錯誤邊界中。如果其中的某些 UI 組件崩潰,其余部分仍然能夠交互。
某些錯誤,錯誤邊界組件不會(huì )捕獲
自身組件的錯誤
異步的錯誤 (如 setTimeout 中拋出的錯誤)
import React, {PureComponent} from 'react' // ErrorBoundary.jsx export default class ErrorBoundary extends PureComponent { state = { isError: false } /* 此函數不會(huì )運行 */ static getDerivedStateFromError(error) { console.log('Rendering Error: ', error) return { isError: true } } render() { if (this.isError) { return <span>Something Wrong...</span> } return this.props.children } } // Comp.jsx Comp 組件 export default funtion Comp() { setTimeout(() => { throw new Error('setTimeout error') }, 1000) return <div>Comp</div> } // App.jsx 使用 export default function App() { return <> <ErrorBoundary> <Comp /> </ErrorBoundary> </> }
事件中拋出的錯誤
即:僅處理渲染子組件期間的同步錯誤
到此這篇關(guān)于React中Portals與錯誤邊界處理實(shí)現的文章就介紹到這了,更多相關(guān)React Portals與錯誤邊界處理內容請搜索腳本之家以前的文章或繼續瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
免責聲明:本站發(fā)布的內容(圖片、視頻和文字)以原創(chuàng )、來(lái)自本網(wǎng)站內容采集于網(wǎng)絡(luò )互聯(lián)網(wǎng)轉載等其它媒體和分享為主,內容觀(guān)點(diǎn)不代表本網(wǎng)站立場(chǎng),如侵犯了原作者的版權,請告知一經(jīng)查實(shí),將立刻刪除涉嫌侵權內容,聯(lián)系我們QQ:712375056,同時(shí)歡迎投稿傳遞力量。
Copyright ? 2009-2022 56dr.com. All Rights Reserved. 特網(wǎng)科技 特網(wǎng)云 版權所有 特網(wǎng)科技 粵ICP備16109289號
域名注冊服務(wù)機構:阿里云計算有限公司(萬(wàn)網(wǎng)) 域名服務(wù)機構:煙臺帝思普網(wǎng)絡(luò )科技有限公司(DNSPod) CDN服務(wù):阿里云計算有限公司 百度云 中國互聯(lián)網(wǎng)舉報中心 增值電信業(yè)務(wù)經(jīng)營(yíng)許可證B2
建議您使用Chrome、Firefox、Edge、IE10及以上版本和360等主流瀏覽器瀏覽本網(wǎng)站