reactjs 为什么我的nextjs组件渲染了两次?

uklbhaso  于 2023-05-17  发布在  React
关注(0)|答案(2)|浏览(555)

这是一个从firebase存储中呈现数据并将其列出的组件。该函数要做的是将从firebase存储中提取的视频设置为useState。这样我就可以调用videos和map到一个新的组件中,它恰好是一个按钮列表。它工作得相对较好,问题是组件渲染了两次,第一次它没有保存状态中的视频,第二次它保存了。换句话说,组件不等待视频被保存在状态中,并且简单地呈现其自身,导致不显示具有视频标题的按钮列表。

// ReactJS
import { useState, useEffect } from "react";

// NextJS
import { useRouter } from "next/router";

// Seo
import Seo from "../../../components/Seo";

// Hooks
import { withProtected } from "../../../hook/route";

// Components
import DashboardLayout from "../../../layouts/Dashboard";

// Firebase
import { getDownloadURL, getMetadata, listAll, ref } from "firebase/storage";
import { storage } from "../../../config/firebase";

// Utils
import capitalize from "../../../utils/capitalize";
import { PlayIcon } from "@heroicons/react/outline";

function Video() {
  // States
  const [videos, setVideos] = useState([]);
  const [videoActive, setVideoActive] = useState(null);

  // Routing
  const router = useRouter();
  const { id } = router.query;

  // Reference
  const reference = ref(storage, `training/${id}`);

  // Check if path is empty

  function getVideos() {
    let items = [];
    listAll(reference).then((res) => {
      res.items.forEach(async (item) => {
        getDownloadURL(ref(storage, `training/${id}/${item.name}`)).then(
          (url) => {
            items.push({
              name: item.name.replace("-", " "),
              href: item.name,
              url,
            });
          }
        );
      });
    });
    setVideos(items);
  }

  useEffect(() => {
    getVideos();
    // eslint-disable-next-line react-hooks/exhaustive-deps
  }, [id]);

  console.log(videos);

  return (
    <>
      <Seo
        title={`${capitalize(id)} Training - Dashboard`}
        description={`${capitalize(
          id
        )} training for all Every Benefits Agents.`}
      />
      <DashboardLayout>
        <h2>{capitalize(reference.name)}</h2>
        <section>
          <video controls controlsList="nodownload">
            {videoActive && <source src={videoActive} type="video/mp4" />}
          </video>
          <ul role="list" className="divide-y divide-gray-200 my-4">
            {videos.map((video, index) => (
              <button key={index} className="py-4 flex">
                <div className="w-full ml-3 flex flex-row justify-start items-center space-x-3">
                  <PlayIcon className="w-6 h-6 text-gray-600" />
                  <p className="text-sm font-medium text-gray-900">
                    {video.name}
                  </p>
                </div>
              </button>
            ))}

            {console.log("Component rendered")}
          </ul>
        </section>
      </DashboardLayout>
    </>
  );
}

export default withProtected(Video);

以下是组件如何开始渲染的示例:

有人知道为什么会这样吗?

brtdzjyr

brtdzjyr1#

我在这个帖子里找到了答案。https://github.com/vercel/next.js/issues/35822
这个问题是关于React 18严格模式的。您可以阅读React 18严格模式中的新内容。
如果你没有使用严格模式,它不应该发生。如果不是很重要,你可以在next.config.js中关闭React Strict Mode。文件如下所示。

const nextConfig = {
  reactStrictMode: false
}

module.exports = nextConfig
rta7y2nd

rta7y2nd2#

这是因为react中的strict模式是用来找出组件级的常见bug的,为了做到这一点,react渲染了两次单个组件。第一个渲染是我们期望的,其他渲染是React严格模式。在开发阶段,您将面临的问题,认为这不是一个问题。这是为了在开发阶段获得更好的开发体验。在生产阶段,如果没有任何故障,默认情况下不会出现此类问题。
我们可以阅读更多:https://react.dev/reference/react/StrictMode
此外,如果这对你来说很烦人,你可以在next.config.js中禁用严格模式**(我个人不喜欢禁用严格模式)**:

const nextConfig = {
  reactStrictMode: false
}

module.exports = nextConfig

谢谢你!

相关问题