重构 React 代码就像把一个混乱的厨房变成一个组织良好的烹饪天堂。它是关于在不改变应用程序功能的情况下改进应用程序的结构、可维护性和性能。无论您是在与臃肿的组件还是混乱的状态逻辑作斗争,精心策划的重构都会将您的代码库转变为流畅、高效的机器。
本博客揭示了常见的重构场景,提供了可行的解决方案,并使您能够释放 React 应用程序的真正潜力。
我。什么是重构以及为什么它很重要?
重构可以在不改变代码功能的情况下改进代码的结构。这不是为了修复错误或添加功能,而是为了让您的代码对人类和机器都更好。
为什么要重构?
- 可读性:当代码读起来像一本好小说而不是一个神秘的谜题时,在凌晨 3 点调试代码会变得容易得多。
- 可维护性:干净的代码库可以节省数小时的入门时间并加快更新速度。
- 性能:更简洁的代码通常意味着更快的加载时间和更流畅的用户体验。
? 专业提示:避免过早优化。当有明确的需求时进行重构,例如改善开发人员体验或解决缓慢的渲染问题。
二.嗅出代码味道
代码异味是低效率或复杂性的微妙信号。它们不是错误,但它们表明了需要改进的地方。
常见的 React 代码味道
-
臃肿的组件
- 问题:单个组件处理太多职责,例如获取数据、渲染和处理事件。
function ProductPage() { const [data, setData] = useState([]); useEffect(() => fetchData(), []); const handleAddToCart = () => { ... }; return ( <div> {data.map(item => <productitem key="{item.id}" item="{item}"></productitem>)} <button onclick="{handleAddToCart}">Add to Cart</button> </div> ); }
- 解决方案: 将其分解为更小的、更集中的组件。
function ProductPage() { return ( <div> <productlist></productlist> <cartbutton></cartbutton> </div> ); } function ProductList() { const [data, setData] = useState([]); useEffect(() => fetchData(), []); return data.map(item => <productitem key="{item.id}" item="{item}"></productitem>); } function CartButton() { const handleAddToCart = () => { ... }; return <button onclick="{handleAddToCart}">Add to Cart</button>; }
-
支柱钻井
- 问题: 通过多层组件传递 props。
<app> <productlist product="{product}"></productlist> </app>
- 解决方案1:使用组合。
<productlist> <productitem product="{product}"></productitem> </productlist>
- 解决方案 2: 使用上下文。
const ProductContext = React.createContext(); function App() { const [product, setProduct] = useState({ id: 1, name: 'Example Product' }); // Example state return ( <productcontext.provider value="{product}"> <productlist></productlist> </productcontext.provider> ); } function ProductList() { const product = useContext(ProductContext); return <productitem product="{product}"></productitem>; }
-
嵌套三元地狱
- 问题: 使用嵌套三元组的复杂条件渲染。
return condition1 ? a : condition2 ? b : condition3 ? c : d;
- 解决方案: 使用辅助函数或 switch 语句进行重构。
function renderContent(condition) { switch (condition) { case 1: return a; case 2: return b; case 3: return c; default: return d; } } return renderContent(condition);
-
重复逻辑
- 问题:跨组件重复相同的逻辑。
function calculateTotal(cart) { return cart.reduce((total, item) => total + item.price, 0); }
- 解决方案: 将共享逻辑移至可重用实用程序或自定义挂钩中。
function calculateTotalPrice(cart) { return cart.reduce((total, item) => total + item.price, 0); } function useTotalPrice(cart) { return useMemo(() => calculateTotalPrice(cart), [cart]); }
-
过度状态
- 问题:直接管理派生状态。
const [isLoggedIn, setIsLoggedIn] = useState(user !== null);
- 解决方案: 使用派生状态代替。
const isLoggedIn = !!user; // Converts 'user' to boolean
三.简化状态管理
状态管理很重要,但很快就会变得混乱。以下是简化方法:
派生状态:计算,不存储
- 问题:存储冗余状态。
- 解决方案:直接从源计算派生值。
const [cartItems, setCartItems] = useState([]); const totalPrice = cartItems.reduce((total, item) => total + item.price, 0);
使用 useReducer 处理复杂状态
- 问题:多个相互依赖的状态。
- 解决方案:使用useReducer。
const initialState = { count: 0 }; function reducer(state, action) { switch (action.type) { case 'increment': return { count: state.count + 1 }; default: return state; } } const [state, dispatch] = useReducer(reducer, initialState);
状态托管
- 问题:用于本地数据的全局状态。
- 解决方案:将状态移近需要的位置。
// Before: function App() { const [filter, setFilter] = useState(''); return <productlist filter="{filter}" onfilterchange="{setFilter}"></productlist>; } // After: function ProductList() { const [filter, setFilter] = useState(''); return <filterinput value="{filter}" onchange="{setFilter}"></filterinput>; }
四。重构组件
组件应该只做一项工作并且做得很好。例如:
每个组件一个作业
function MemberCard({ member }) { return ( <div> <summary member="{member}"></summary> <seemore details="{member.details}"></seemore> </div> ); }
V.性能优化
反应分析器
使用分析器来识别瓶颈。在“Profiler”下的开发者工具中访问它。
记忆
优化昂贵的计算:
function ProductPage() { const [data, setData] = useState([]); useEffect(() => fetchData(), []); const handleAddToCart = () => { ... }; return ( <div> {data.map(item => <productitem key="{item.id}" item="{item}"></productitem>)} <button onclick="{handleAddToCart}">Add to Cart</button> </div> ); }
注意:避免过度使用频繁更新的依赖项的记忆。
六。重构可测试性
编写以用户为中心的测试:
function ProductPage() { return ( <div> <productlist></productlist> <cartbutton></cartbutton> </div> ); } function ProductList() { const [data, setData] = useState([]); useEffect(() => fetchData(), []); return data.map(item => <productitem key="{item.id}" item="{item}"></productitem>); } function CartButton() { const handleAddToCart = () => { ... }; return <button onclick="{handleAddToCart}">Add to Cart</button>; }
七。可维护性的最后润色
- 按功能组织:
<app> <productlist product="{product}"></productlist> </app>
- 使用绝对导入:
<productlist> <productitem product="{product}"></productitem> </productlist>
八。备忘单
Category | Tip |
---|---|
Code Smells | Split bloated components; avoid prop drilling. |
State Management | Use derived state; colocate state. |
Performance | Use Profiler; optimize Context values. |
Testing | Test behavior, not implementation details. |
提示
以上是重构 React:驯服混乱,一次一个组件的详细内容。更多信息请关注PHP中文网其他相关文章!

JavaScript字符串替换方法详解及常见问题解答 本文将探讨两种在JavaScript中替换字符串字符的方法:在JavaScript代码内部替换和在网页HTML内部替换。 在JavaScript代码内部替换字符串 最直接的方法是使用replace()方法: str = str.replace("find","replace"); 该方法仅替换第一个匹配项。要替换所有匹配项,需使用正则表达式并添加全局标志g: str = str.replace(/fi

利用轻松的网页布局:8个基本插件 jQuery大大简化了网页布局。 本文重点介绍了简化该过程的八个功能强大的JQuery插件,对于手动网站创建特别有用

因此,在这里,您准备好了解所有称为Ajax的东西。但是,到底是什么? AJAX一词是指用于创建动态,交互式Web内容的一系列宽松的技术。 Ajax一词,最初由Jesse J创造

本教程演示了创建通过Ajax加载的动态页面框,从而可以即时刷新,而无需全页重新加载。 它利用jQuery和JavaScript。将其视为自定义的Facebook式内容框加载程序。 关键概念: Ajax和JQuery

10款趣味横生的jQuery游戏插件,让您的网站更具吸引力,提升用户粘性!虽然Flash仍然是开发休闲网页游戏的最佳软件,但jQuery也能创造出令人惊喜的效果,虽然无法与纯动作Flash游戏媲美,但在某些情况下,您也能在浏览器中获得意想不到的乐趣。 jQuery井字棋游戏 游戏编程的“Hello world”,现在有了jQuery版本。 源码 jQuery疯狂填词游戏 这是一个填空游戏,由于不知道单词的上下文,可能会产生一些古怪的结果。 源码 jQuery扫雷游戏

此JavaScript库利用窗口。名称属性可以管理会话数据,而无需依赖cookie。 它为浏览器中存储和检索会话变量提供了强大的解决方案。 库提供了三种核心方法:会话

本教程演示了如何使用jQuery创建迷人的视差背景效果。 我们将构建一个带有分层图像的标题横幅,从而创造出令人惊叹的视觉深度。 更新的插件可与JQuery 1.6.4及更高版本一起使用。 下载


热AI工具

Undresser.AI Undress
人工智能驱动的应用程序,用于创建逼真的裸体照片

AI Clothes Remover
用于从照片中去除衣服的在线人工智能工具。

Undress AI Tool
免费脱衣服图片

Clothoff.io
AI脱衣机

AI Hentai Generator
免费生成ai无尽的。

热门文章

热工具

EditPlus 中文破解版
体积小,语法高亮,不支持代码提示功能

安全考试浏览器
Safe Exam Browser是一个安全的浏览器环境,用于安全地进行在线考试。该软件将任何计算机变成一个安全的工作站。它控制对任何实用工具的访问,并防止学生使用未经授权的资源。

Dreamweaver CS6
视觉化网页开发工具

SublimeText3 Linux新版
SublimeText3 Linux最新版

mPDF
mPDF是一个PHP库,可以从UTF-8编码的HTML生成PDF文件。原作者Ian Back编写mPDF以从他的网站上“即时”输出PDF文件,并处理不同的语言。与原始脚本如HTML2FPDF相比,它的速度较慢,并且在使用Unicode字体时生成的文件较大,但支持CSS样式等,并进行了大量增强。支持几乎所有语言,包括RTL(阿拉伯语和希伯来语)和CJK(中日韩)。支持嵌套的块级元素(如P、DIV),