0

0

React 重新渲染深度解析:为何 children 组件会被重复渲染及优化策略

花韻仙語

花韻仙語

发布时间:2025-10-06 12:05:21

|

853人浏览过

|

来源于php中文网

原创

react 重新渲染深度解析:为何 children 组件会被重复渲染及优化策略

本文深入探讨了 React 组件在父组件状态更新时,即使通过 children prop 传递,子组件仍可能被重复渲染的常见问题。核心原因在于父组件每次渲染时,若子组件在 JSX 中被内联声明,React 会创建新的子组件实例。文章通过具体代码示例,详细解释了这一机制,并提供了将状态管理下移以稳定 children prop 的优化方案,旨在帮助开发者更好地理解和优化 React 应用的渲染性能。

理解 React 组件的重新渲染机制

在 React 应用开发中,组件的重新渲染(re-render)是一个核心概念,但其行为有时会超出开发者的预期。一个常见的误解是,如果一个父组件的 children prop 逻辑上没有改变,那么传递给它的子组件就不会重新渲染。然而,当父组件自身因状态更新而重新渲染时,如果子组件是在父组件的 JSX 渲染逻辑中内联声明的,即使其内容看似不变,React 也会将其视为一个新的组件实例,从而触发子组件的重新渲染。

这背后的机制是:每当一个组件函数执行时(即发生渲染),如果其 JSX 中包含对另一个组件的引用,React 会为该引用创建一个新的 React 元素(React Element)。即使这个新的 React 元素在类型和 props 上与上一次渲染的元素完全相同,但由于它是一个“新”的对象实例,React 在协调(reconciliation)过程中会认为这个子组件可能需要更新,并会访问其子树进行比对,最终导致子组件函数被再次调用,即重新渲染。

问题示例:children 组件的意外重新渲染

考虑以下 React 应用结构,其中 App 组件包含一个定时器,每 100 毫秒更新一次自身状态,进而导致 App 组件重新渲染。App 将 Child 组件作为 Parent 组件的 children prop 传递。

import { useState, useEffect } from 'react';

// Child 组件,每次渲染都会在控制台输出 'rendered'
const Child = () => {
  console.log('Child rendered');
  return (
    <p>这是一个子组件内容。</p>
  );
}

// Parent 组件,接收并渲染 children prop
const Parent = ({ children }) => {
  return (
    <div id='parent'>
      {children}
    </div>
  );
}

// App 组件,包含一个定时器更新自身状态
const App = () => {
  const [now, setNow] = useState();

  // 启动一个定时器,每100ms更新 'now' 状态
  useEffect(() => {
    const interval = setInterval(() =>
      setNow(Date.now()), 100);
    return () => clearInterval(interval); // 清理定时器
  }, []);

  return (
    <div className="App">
      <Parent>
        {/* Child 组件在这里被内联声明 */}
        <Child />
      </Parent>
    </div>
  );
}

export default App;

运行上述代码,你会发现控制台每 100 毫秒都会输出 Child rendered。尽管 Child 组件没有任何自身状态或接收任何 props,并且 Parent 组件也只是简单地渲染其 children prop,Child 依然在 App 组件状态更新时被重复渲染。

原因分析:

当 App 组件的状态 now 更新时,App 组件会重新执行其渲染函数。在每次执行时,App 组件的 JSX 表达式 都会被重新评估。这意味着,每次 App 渲染时,都会创建一个新的 React 元素实例。

尽管这个新的 元素与上一次渲染的 元素在类型和结构上是相同的,但对 React 而言,它是一个全新的 JavaScript 对象引用。因此,当 App 将这个新的 元素作为 Parent 的 children prop 传递时,React 会认为 Parent 组件的 children prop 已经改变(因为它是一个新的对象引用),从而触发 Parent 和其 children(即 Child 组件)的重新渲染。

优化方案:稳定化 children Prop

为了避免 Child 组件在 App 组件状态更新时进行不必要的重新渲染,我们需要确保传递给 Parent 组件的 children prop 在 App 重新渲染时保持引用稳定。最直接有效的方法是将导致重新渲染的状态或副作用下移到组件树中更低层级的组件,使其不会影响到不相关的上层组件或兄弟组件。

在我们的例子中,App 组件中的定时器状态更新是导致 Child 重新渲染的根本原因。如果我们将这个定时器逻辑移动到 Parent 组件内部,那么 App 组件将不再因定时器而重新渲染,从而稳定了传递给 Parent 的 children prop。

Stable Diffusion 2.1 Demo
Stable Diffusion 2.1 Demo

最新体验版 Stable Diffusion 2.1

下载
import { useState, useEffect } from 'react';

const Child = () => {
  console.log('Child rendered');
  return (
    <p>这是一个子组件内容。</p>
  );
}

// Parent 组件,现在包含了定时器状态
const Parent = ({ children }) => {
  const [now, setNow] = useState(); // 状态移至 Parent

  // 启动一个定时器,每100ms更新 'now' 状态
  useEffect(() => {
    const interval = setInterval(() =>
      setNow(Date.now()), 100);
    return () => clearInterval(interval);
  }, []);

  return (
    <div id='parent'>
      {children}
    </div>
  );
}

// App 组件现在不包含定时器逻辑
const App = () => {
  return (
    <div className="App">
      <Parent>
        {/* Child 组件仍然在这里被内联声明 */}
        <Child />
      </Parent>
    </div>
  );
}

export default App;

通过将 useState 和 useEffect 钩子从 App 组件移动到 Parent 组件,App 组件在首次渲染后,其内部将不再有状态更新导致自身重新渲染。这意味着 这部分 JSX 表达式只会在 App 组件挂载时执行一次。因此,传递给 Parent 的 React 元素引用将保持稳定。

现在,Parent 组件会每 100 毫秒更新其内部的 now 状态并重新渲染。然而,由于 App 组件不再重新渲染,它传递给 Parent 的 children prop(即 元素)的引用始终是第一次创建的那个。当 Parent 重新渲染时,React 发现其 children prop 的引用没有改变,因此不会访问 Child 组件的子树,Child 组件也就不会重新渲染。此时,控制台将只输出一次 Child rendered。

注意事项与最佳实践

  1. 状态下移原则: 尽可能将状态和副作用放置在组件树中需要它们的最底层组件。这可以有效限制重新渲染的范围,避免不必要的性能开销。

  2. React.memo 的作用: 对于功能组件,可以使用 React.memo 来进行性能优化。React.memo 会对组件的 props 进行浅比较,如果 props 没有改变,则跳过组件的重新渲染。在上述示例中,即使 App 仍然包含定时器,如果 Child 组件被 React.memo 包裹,并且它不接收任何 props,它将不会重新渲染。但请注意,React.memo 仅在 props 引用稳定时才有效。如果像原始问题中那样,每次父组件渲染都创建一个新的 Child 元素,即使 Child 被 memo 包裹,它仍然会接收到一个“新”的 children prop 引用,从而导致重新渲染。

    const MemoizedChild = React.memo(() => {
      console.log('MemoizedChild rendered');
      return <p>这是一个被 memoized 的子组件内容。</p>;
    });
    
    // ... 在 App 中使用 <MemoizedChild />

    然而,在我们的原始问题场景中,Child 是作为 Parent 的 children prop 传递的。如果 Parent 被 React.memo 包裹,而 App 每次都传递一个新的 元素给 Parent 的 children prop,那么 Parent 的 children prop 引用仍然会改变,Parent 依然会重新渲染,并进而渲染其 children。因此,React.memo 主要用于优化组件自身因父组件 props 变化而导致的重新渲染,而不是解决父组件内联创建子组件实例的问题。

  3. 理解 JSX 的本质: JSX 语法糖最终会被编译成 React.createElement() 调用。每次父组件渲染时,JSX 中的组件标签都会被转换为新的 React.createElement() 调用,生成新的 React 元素对象。理解这一点有助于避免关于组件重新渲染的误解。

  4. 组件组合的权衡: “提升内容” (lifting content up) 是一种强大的组合模式,可以将子组件的渲染逻辑与父组件的业务逻辑解耦。但在使用时,需要注意其对渲染性能的影响,确保传递的 children prop 引用在不必要时是稳定的。

总结

React 的渲染机制是其高性能的基础,但如果不深入理解其工作原理,可能会遇到一些性能陷阱。当父组件重新渲染且在 JSX 中内联声明子组件时,React 会创建新的子组件实例,即使其逻辑内容不变,也可能导致不必要的重新渲染。通过将状态和副作用下移到更合适的组件层级,我们可以有效稳定传递给子组件的 children prop 引用,从而避免这些不必要的重新渲染,提升应用的性能和用户体验。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
PHP 高并发与性能优化
PHP 高并发与性能优化

本专题聚焦 PHP 在高并发场景下的性能优化与系统调优,内容涵盖 Nginx 与 PHP-FPM 优化、Opcode 缓存、Redis/Memcached 应用、异步任务队列、数据库优化、代码性能分析与瓶颈排查。通过实战案例(如高并发接口优化、缓存系统设计、秒杀活动实现),帮助学习者掌握 构建高性能PHP后端系统的核心能力。

111

2025.10.16

PHP 数据库操作与性能优化
PHP 数据库操作与性能优化

本专题聚焦于PHP在数据库开发中的核心应用,详细讲解PDO与MySQLi的使用方法、预处理语句、事务控制与安全防注入策略。同时深入分析SQL查询优化、索引设计、慢查询排查等性能提升手段。通过实战案例帮助开发者构建高效、安全、可扩展的PHP数据库应用系统。

99

2025.11.13

JavaScript 性能优化与前端调优
JavaScript 性能优化与前端调优

本专题系统讲解 JavaScript 性能优化的核心技术,涵盖页面加载优化、异步编程、内存管理、事件代理、代码分割、懒加载、浏览器缓存机制等。通过多个实际项目示例,帮助开发者掌握 如何通过前端调优提升网站性能,减少加载时间,提高用户体验与页面响应速度。

36

2025.12.30

JavaScript浏览器渲染机制与前端性能优化实践
JavaScript浏览器渲染机制与前端性能优化实践

本专题围绕 JavaScript 在浏览器中的执行与渲染机制展开,系统讲解 DOM 构建、CSSOM 解析、重排与重绘原理,以及关键渲染路径优化方法。内容涵盖事件循环机制、异步任务调度、资源加载优化、代码拆分与懒加载等性能优化策略。通过真实前端项目案例,帮助开发者理解浏览器底层工作原理,并掌握提升网页加载速度与交互体验的实用技巧。

78

2026.03.06

JavaScript浏览器渲染机制与前端性能优化实践
JavaScript浏览器渲染机制与前端性能优化实践

本专题围绕 JavaScript 在浏览器中的执行与渲染机制展开,系统讲解 DOM 构建、CSSOM 解析、重排与重绘原理,以及关键渲染路径优化方法。内容涵盖事件循环机制、异步任务调度、资源加载优化、代码拆分与懒加载等性能优化策略。通过真实前端项目案例,帮助开发者理解浏览器底层工作原理,并掌握提升网页加载速度与交互体验的实用技巧。

59

2026.03.06

Rust内存安全机制与所有权模型深度实践
Rust内存安全机制与所有权模型深度实践

本专题围绕 Rust 语言核心特性展开,深入讲解所有权机制、借用规则、生命周期管理以及智能指针等关键概念。通过系统级开发案例,分析内存安全保障原理与零成本抽象优势,并结合并发场景讲解 Send 与 Sync 特性实现机制。帮助开发者真正理解 Rust 的设计哲学,掌握在高性能与安全性并重场景中的工程实践能力。

148

2026.03.05

PHP高性能API设计与Laravel服务架构实践
PHP高性能API设计与Laravel服务架构实践

本专题围绕 PHP 在现代 Web 后端开发中的高性能实践展开,重点讲解基于 Laravel 框架构建可扩展 API 服务的核心方法。内容涵盖路由与中间件机制、服务容器与依赖注入、接口版本管理、缓存策略设计以及队列异步处理方案。同时结合高并发场景,深入分析性能瓶颈定位与优化思路,帮助开发者构建稳定、高效、易维护的 PHP 后端服务体系。

273

2026.03.04

AI安装教程大全
AI安装教程大全

2026最全AI工具安装教程专题:包含各版本AI绘图、AI视频、智能办公软件的本地化部署手册。全篇零基础友好,附带最新模型下载地址、一键安装脚本及常见报错修复方案。每日更新,收藏这一篇就够了,让AI安装不再报错!

93

2026.03.04

Swift iOS架构设计与MVVM模式实战
Swift iOS架构设计与MVVM模式实战

本专题聚焦 Swift 在 iOS 应用架构设计中的实践,系统讲解 MVVM 模式的核心思想、数据绑定机制、模块拆分策略以及组件化开发方法。内容涵盖网络层封装、状态管理、依赖注入与性能优化技巧。通过完整项目案例,帮助开发者构建结构清晰、可维护性强的 iOS 应用架构体系。

159

2026.03.03

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
React 教程
React 教程

共58课时 | 5.9万人学习

国外Web开发全栈课程全集
国外Web开发全栈课程全集

共12课时 | 1万人学习

React核心原理新老生命周期精讲
React核心原理新老生命周期精讲

共12课时 | 1.1万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号