首页 > web前端 > js教程 > 正文

React 组件参数未更新导致数据未刷新问题的解决方案

花韻仙語
发布: 2025-10-08 10:50:01
原创
939人浏览过

react 组件参数未更新导致数据未刷新问题的解决方案

本文旨在解决 React 应用中,父组件向子组件传递参数后,子组件未能根据新的参数值及时更新数据的问题。通过分析问题代码,我们将定位到表单提交导致的页面刷新是罪魁祸首,并提供使用 e.preventDefault() 阻止默认行为的解决方案,确保组件能够正确响应参数变化并刷新数据。

在 React 应用开发中,父子组件之间的数据传递是常见的模式。然而,有时我们可能会遇到这样的问题:父组件传递给子组件的参数发生了变化,但子组件并没有根据新的参数值重新渲染或获取数据。这会导致 UI 显示的数据与预期不符,影响用户体验。

下面我们通过一个实际案例来分析这个问题,并提供解决方案。

问题描述

一个应用包含一个搜索栏(Navbar)和一个数据表格(AttacksGrid)。用户在搜索栏中输入关键词,点击搜索按钮后,关键词会被传递给 AttacksGrid 组件,AttacksGrid 组件会根据关键词从后端获取数据并显示在表格中。

然而,实际情况是,每次点击搜索按钮后,AttacksGrid 组件都会显示空数据,或者显示初始状态的数据,并没有根据新的关键词进行数据刷新。

代码分析

以下是相关组件的代码:

App 组件:

function App() {
  const [query, setQuery] = useState("");

  const searchClicked = (keyword) => {
    console.log(`setting query to: ${keyword}`);
    setQuery(keyword); // for now we want to search for the keyword in the name and description, platform is not supported yet
  };

  return (
    <div>
      <Navbar onSearchClicked={searchClicked} />
      <AttacksGrid query={query} />;
    </div>
  );
}
登录后复制

AttacksGrid 组件:

function AttacksGrid({ query }) {
  console.log(`QUERY IS ${query}`);
  useEffect(() => {
    fetch(`http://127.0.0.1:9000/attacks?name=${query}&description=${query}`)
      .then((response) => response.json())
      .then((data) => {
        console.log(data);
        setAttacks(data);
      })
      .catch((err) => console.log(err));
  }, [query]);

  const [attacks, setAttacks] = useState([]);

  return (
    <>
      <table className="table table-dark">
        <thead>
          <tr>
            <th scope="col">Name</th>
            <th scope="col">Description</th>
            <th scope="col">Platforms</th>
            <th scope="col">Detection</th>
            <th scope="col">Phase</th>
          </tr>
        </thead>

        {attacks.map((attack) => (
          <tbody key={attack.ID}>
            <tr>
              <th scope="row">{attack.NAME}</th>
              <td>{attack.DESCRIPTION}</td>
              <td>{attack.X_MITRE_PLATFORMS}</td>
              <td>{attack.X_MITRE_DETECTION}</td>
              <td>{attack.PHASE_NAME}</td>
            </tr>
          </tbody>
        ))}
      </table>
    </>
  );
}
登录后复制

Navbar 组件:

AI新媒体文章
AI新媒体文章

专为新媒体人打造的AI写作工具,提供“选题创作”、“文章重写”、“爆款标题”等功能

AI新媒体文章75
查看详情 AI新媒体文章
function Navbar({ onSearchClicked }) {
  const [message, setMessage] = useState("");
  const handleChange = (event) => {
    setMessage(event.target.value);
    console.log(message);
  };
  return (
    <nav className="navbar bg-body-tertiary">
      <div className="container-fluid">
        <form className="d-flex" role="search">
          <input
            id="message"
            name="message"
            className="form-control me-2"
            type="search"
            placeholder="Search"
            aria-label="Search"
            onChange={handleChange}
          />
          <button
            className="btn btn-outline-success"
            type="submit"
            onClick={() => {
              onSearchClicked(message);
            }}
          >
            Search
          </button>
        </form>
      </div>
    </nav>
  );
}
登录后复制

通过观察代码,我们发现 AttacksGrid 组件的 useEffect 依赖于 query 变量,当 query 发生变化时,useEffect 应该重新执行,从而从后端获取新的数据。

问题出在 Navbar 组件中的表单提交。当点击搜索按钮时,由于 button 的 type 属性为 submit,表单会默认提交,导致页面刷新。页面刷新后,App 组件的状态会被重置,query 重新变为空字符串,因此 AttacksGrid 组件总是显示初始状态的数据。

解决方案

要解决这个问题,我们需要阻止表单的默认提交行为。可以使用 e.preventDefault() 方法来实现。

修改 Navbar 组件中的 button 的 onClick 事件处理函数:

<button
  className="btn btn-outline-success"
  type="submit"
  onClick={(e) => {
   e.preventDefault(); //Preventing default page refresh submit form action
   onSearchClicked(message);
  }}
>
  Search
</button>
登录后复制

通过在 onClick 事件处理函数中调用 e.preventDefault(),我们可以阻止表单的默认提交行为,从而避免页面刷新。

完整代码

以下是修改后的 Navbar 组件的完整代码:

function Navbar({ onSearchClicked }) {
  const [message, setMessage] = useState("");
  const handleChange = (event) => {
    setMessage(event.target.value);
    console.log(message);
  };
  return (
    <nav className="navbar bg-body-tertiary">
      <div className="container-fluid">
        <form className="d-flex" role="search">
          <input
            id="message"
            name="message"
            className="form-control me-2"
            type="search"
            placeholder="Search"
            aria-label="Search"
            onChange={handleChange}
          />
          <button
            className="btn btn-outline-success"
            type="submit"
            onClick={(e) => {
              e.preventDefault(); //Preventing default page refresh submit form action
              onSearchClicked(message);
            }}
          >
            Search
          </button>
        </form>
      </div>
    </nav>
  );
}
登录后复制

总结

在 React 应用中,如果子组件未能根据父组件传递的参数及时更新数据,需要检查以下几点:

  1. 父组件是否正确地更新了状态。
  2. 子组件的 useEffect 是否正确地依赖于父组件传递的参数。
  3. 是否存在表单提交或其他导致页面刷新的操作。

通过使用 e.preventDefault() 阻止表单的默认提交行为,我们可以避免页面刷新,确保组件能够正确响应参数变化并刷新数据.

以上就是React 组件参数未更新导致数据未刷新问题的解决方案的详细内容,更多请关注php中文网其它相关文章!

最佳 Windows 性能的顶级免费优化软件
最佳 Windows 性能的顶级免费优化软件

每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。

下载
来源:php中文网
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn
最新问题
开源免费商场系统广告
热门教程
更多>
最新下载
更多>
网站特效
网站源码
网站素材
前端模板
关于我们 免责申明 意见反馈 讲师合作 广告合作 最新更新 English
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送
PHP中文网APP
随时随地碎片化学习
PHP中文网抖音号
发现有趣的

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