记一个React组件入参不当导致页面卡死的问题

一、问题描述

1.1 触发现象

点击按钮后页面卡死

1.2 最小 Demo

javascript 复制代码
import './App.css';
import React, { useState, useEffect } from "react";

const Demo = ({ value = [] }) => {
  const [state, setState] = useState();
  useEffect(() => {
    console.log("value", value);
    setState((value || []).filter((item) => item !== ""));
  }, [value]);

  return <div>list:{state}</div>;
};

export default function App() {
  const [list, setList] = useState(["1", "2"]);
  return (
    <div className="App">
      <Demo value={list} />
      <h1 onClick={() => setList([...list, "a"])}>Add List</h1>
      <h1 onClick={() => setList(undefined)}>Clear List</h1>
    </div>
  );
}

二、原因分析

2.1 排查过程

2.1.1 console 输出查看

没有报错日志

2.1.2 performance 查看

setState 方法耗时较长

2.1.3 源码屏蔽分析

二分法屏蔽问题代码并大致定位范围为 Demo 组件引起

2.1.4 源码加 log 分析

第 7 行加 log 发现,Value 入参传为 undefined 时会循环打印 log

2.2 原因分析

  1. 入参默认空数组不合理,组件内部更新状态都会拿到一个全新的入参空数组
javascript 复制代码
const Demo = ({ value = [] }) => {
  1. 状态更新不合理,依赖入参状态并处理后再显示,还使用了空数组兜底,这里也没有判断入参 Value 本身为空的情况
javascript 复制代码
  useEffect(() => {
    console.log("value", value);
    setState((value || []).filter((item) => item !== ""));
  }, [value]);

三、后续预防

3.1 入参默认值

使用 useEffect 监听的入参尽量不给默认值,并且处理好入参的各种边界情况

3.2 入参与视图

简单的入参处理尽量省略,可以直接使用入参做视图展示,比如:

javascript 复制代码
  return <div>list:{value?.length > 0 && value.filter(Boolean)}</div>;
相关推荐
qq_177767372 分钟前
React Native鸿蒙跨平台数据使用监控应用技术,通过setInterval每5秒更新一次数据使用情况和套餐使用情况,模拟了真实应用中的数据监控场景
开发语言·前端·javascript·react native·react.js·ecmascript·harmonyos
烬头88215 分钟前
React Native鸿蒙跨平台应用实现了onCategoryPress等核心函数,用于处理用户交互和状态更新,通过计算已支出和剩余预算
前端·javascript·react native·react.js·ecmascript·交互·harmonyos
2601_949593652 小时前
基础入门 React Native 鸿蒙跨平台开发:卡片组件
react native·react.js·harmonyos
天人合一peng2 小时前
Unity中button 和toggle监听事件函数有无参数
前端·unity·游戏引擎
方也_arkling3 小时前
别名路径联想提示。@/统一文件路径的配置
前端·javascript
毕设源码-朱学姐3 小时前
【开题答辩全过程】以 基于web教师继续教育系统的设计与实现为例,包含答辩的问题和答案
前端
qq_177767373 小时前
React Native鸿蒙跨平台剧集管理应用实现,包含主应用组件、剧集列表、分类筛选、搜索排序等功能模块
javascript·react native·react.js·交互·harmonyos
qq_177767374 小时前
React Native鸿蒙跨平台自定义复选框组件,通过样式数组实现选中/未选中状态的样式切换,使用链式调用替代样式数组,实现状态驱动的样式变化
javascript·react native·react.js·架构·ecmascript·harmonyos·媒体
web打印社区4 小时前
web-print-pdf:突破浏览器限制,实现专业级Web静默打印
前端·javascript·vue.js·electron·html
RFCEO4 小时前
前端编程 课程十三、:CSS核心基础1:CSS选择器
前端·css·css基础选择器详细教程·css类选择器使用方法·css类选择器命名规范·css后代选择器·精准选中嵌套元素