记一个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>;
相关推荐
山河木马几秒前
前端学习C++之:.h(.hpp)与.cpp文件
前端·javascript·c++
用户9272472502191 分钟前
PHP + CSS + JS + JSON 数据采集与展示系统,支持伪静态
前端
努力只为躺平5 分钟前
一文搞懂 Promise 并发控制:批量执行 vs 最大并发数,实用场景全解析!
前端·javascript
李大玄7 分钟前
Google浏览器拓展工具 "GU"->google Utils
前端·javascript·github
爱编程的喵8 分钟前
从DOM0到事件委托:揭秘JavaScript事件机制的性能密码
前端·javascript·dom
蓝倾13 分钟前
京东批量获取商品SKU操作指南
前端·后端·api
JSLove20 分钟前
常见 npm 报错问题
前端·npm
sunbyte20 分钟前
50天50个小项目 (Vue3 + Tailwindcss V4) ✨ | ContentPlaceholder(背景占位)
前端·javascript·css·vue.js·tailwindcss
爱学习的茄子21 分钟前
React Hooks进阶:从0到1打造高性能Todo应用
前端·react.js·面试
知性的小mahua25 分钟前
vue3+canvas实现摄像头ROI区域标记
前端