记一个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>;
相关推荐
糖墨夕3 分钟前
Trae还能将Figma 设计稿转化为前端代码
前端·trae
程序猿小D4 分钟前
第26节 Node.js 事件
服务器·前端·javascript·node.js·编辑器·ecmascript·vim
天天打码5 分钟前
Bootstrap Table开源的企业级数据表格集成
前端·开源·bootstrap
Allen Bright7 分钟前
【CSS-8】深入理解CSS选择器权重:掌握样式优先级的关键
前端·css
hnlucky10 分钟前
安装vue的教程——Windows Node.js Vue项目搭建
前端·javascript·vue.js·windows·node.js
余道各努力,千里自同风23 分钟前
CSS“多列布局”
前端·css·html
Keya29 分钟前
使用 tinypng 脚本打包为exe 进行压缩图片
前端·python·程序员
wordbaby35 分钟前
React Router 的 handle 和 useMatches 的作用、场景和联系
前端·react.js
我的div丢了肿么办35 分钟前
ResizeObserver和IntersectionObserver的详细讲解
前端·javascript·vue.js
凌览35 分钟前
斩获 7k Star,这个桌宠项目火了🔥
前端·javascript·后端