iis服务器助手广告广告
返回顶部
首页 > 资讯 > 精选 >React Hooks与setInterval的坑怎么解决
  • 191
分享到

React Hooks与setInterval的坑怎么解决

2023-06-30 11:06:28 191人浏览 安东尼
摘要

这篇文章主要讲解了“React Hooks与setInterval的坑怎么解决”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“React Hooks与setInterval

这篇文章主要讲解了“React Hooks与setInterval的坑怎么解决”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“React Hooks与setInterval的坑怎么解决”吧!

一、需求

我们希望有一个每一秒自动+1的定时器

function Counter() {  let [count, setCount] = useState(0);  useEffect(() => {    let id = setInterval(() => {      setCount(count + 1);    }, 1000);    return () => clearInterval(id);  }, [count]);  return <h2>{count}</h2>;}

这种写法你会发现页面效果确实能出来,但是性能很差。每当 count 更改了, useEffect 就会渲染一次,定时器也会不停的被新增与移除。过程如下:

//第一次function Counter() {//... useEffect(() => {    let id = setInterval(() => {      setCount(0 + 1);    }, 1000);    return () => clearInterval(id);  }, [0]);//...}//第二次function Counter() {//... useEffect(() => {    let id = setInterval(() => {      setCount(1 + 1);    }, 1000);    return () => clearInterval(id);  }, [1]);//...//第N次}

现在我们的需求是在实现功能的基础上,还要使得定时器只监听一次,保障它的性能很高。

二、解决方案

1、函数式更新

useState 中的set方法可接收函数,该函数将接收先前的 state ,并返回一个更新后的值。这样定时器每次拿到的是最新的值。

function Counter() {let [count, setCount] = useState(0);useEffect(() => {    let id = setInterval(() => {      setCount(v => {        return v + 1;      });    }, 1000);    return () => clearInterval(id);  }, []);return <h2>{count}</h2>;}

2、使用useRef

useRef 返回一个可变的 ref 对象,返回的 ref 对象在组件的整个生命周期内保持不变。

将定时器函数提取出来,每次定时器触发时,都能取到最新到 count .

function Counter() {  let [count, setCount] = useState(0);  const myRef = useRef(null);  myRef.current = () => {    setCount(count + 1);  };  useEffect(() => {    let id = setInterval(() => {      myRef.current();    }, 1000);    return () => clearInterval(id);  }, []);  return <h2>{count}</h2>;}

思考:为什么不直接像下面这个例子,将setInterval写成 setInterval(myRef.current, 1000)这样呢?为什么要通过一个函数返回?

//这个例子是错误的function Counter() {  let [count, setCount] = useState(0);  const myRef = useRef(null);  myRef.current = () => {    setCount(count + 1);  };  useEffect(() => {    let id = setInterval(myRef.current, 1000);    return () => clearInterval(id);  }, []); return <h2>{count}</h2>;}

定时器的第一个参数为 interval 变量,如果直接将myRef.current直接赋值给 interval 变量,那么之后的myRef.current的值改变之后,在这里依旧取到的是改变之前的值,因为ref的改变不会引起组件的重新渲染

3、用useReducer

将 count 变量存入 reducer 中,使用 useReducer 更新 count

function reducer(state, action) {  switch (action.type) {    case "increment":      return state + 1;    default:      throw new Error();  }}function Counter() {  const [state, dispatch] = useReducer(reducer, 0);  useEffect(() => {    setInterval(() => {      dispatch({ type: "increment" });    }, 1000);  }, []);  return <h2>{state}</h2>;}

4、自定义的hooks

自定义hook:useInterval

import React, { useState, useEffect, useRef } from 'react'; function useInterval(callback, delay) {  const savedCallback = useRef();   // 保存新回调  useEffect(() => {    savedCallback.current = callback;  });   // 建立 interval  useEffect(() => {    function tick() {      savedCallback.current();    }    if (delay !== null) {      let id = setInterval(tick, delay);      return () => clearInterval(id);    }  }, [delay]);}

使用useInterval

function Counter() {  let [count, setCount] = useState(0);   useInterval(() => {    // 你自己的代码    setCount(count + 1);  }, 1000);   return <h2>{count}</h2>;}

useInterval这个api的设计是非常巧妙的。

  • 首先useIntervalsetInterval接收的参数是一样的,这就降低了我们的学习成本

  • 其次,useInterval的delay参数是可以动态调整的,而setInterval的delay参数是没有办法动态调整的

    • useInterval Hook 接收到不同 delay,它会重设 interval。

    • 声明一个带有动态调整 delay 的 interval,来替代写 添加和清除* interval 的代码 &mdash;&mdash; useInterval Hook 帮我们做到了**。

    • 如果想要暂时暂停 interval ,那么可以像下面这个例子一样

  const [delay, setDelay] = useState(1000);  const [isRunning, setIsRunning] = useState(true);   useInterval(() => {    setCount(count + 1);  }, isRunning ? delay : null);
  • useInterval的delay也可以受控于另外一个useInterval

function Counter() {  const [delay, setDelay] = useState(1000);  const [count, setCount] = useState(0);   // 增加计数器  useInterval(() => {    setCount(count + 1);  }, delay);   // 每秒加速  useInterval(() => {    if (delay > 10) {      setDelay(delay / 2);    }  }, 1000);   function handleReset() {    setDelay(1000);  }   return (    <>      <h2>Counter: {count}</h2>      <h5>Delay: {delay}</h5>      <button onClick={handleReset}>        Reset delay      </button>    </>  );}

感谢各位的阅读,以上就是“React Hooks与setInterval的坑怎么解决”的内容了,经过本文的学习后,相信大家对React Hooks与setInterval的坑怎么解决这一问题有了更深刻的体会,具体使用情况还需要大家实践验证。这里是编程网,小编将为大家推送更多相关知识点的文章,欢迎关注!

--结束END--

本文标题: React Hooks与setInterval的坑怎么解决

本文链接: https://www.lsjlt.com/news/328303.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • React Hooks与setInterval的坑怎么解决
    这篇文章主要讲解了“React Hooks与setInterval的坑怎么解决”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“React Hooks与setInterval...
    99+
    2023-06-30
  • react中hooks解决的问题有哪些
    这篇文章主要介绍“react中hooks解决的问题有哪些”,在日常操作中,相信很多人在react中hooks解决的问题有哪些问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”re...
    99+
    2024-04-02
  • 怎么理解React hooks的渲染逻辑
    这篇文章主要介绍“怎么理解React hooks的渲染逻辑”,在日常操作中,相信很多人在怎么理解React hooks的渲染逻辑问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”...
    99+
    2024-04-02
  • 怎么构建自己的react hooks
    这篇文章主要介绍怎么构建自己的react hooks,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!1. 常用的一个 hooks官方中提供了几个内置的钩子,我们简单了解下他们的用法。1.1 useState: 状态钩子...
    99+
    2023-06-15
  • React Hooks在React-refresh模块热替换下的异常行为怎么解决
    本篇内容主要讲解“React Hooks在React-refresh模块热替换下的异常行为怎么解决”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“React Ho...
    99+
    2024-04-02
  • 怎么使用不同的React hooks来解决日常所遇到的问题
    这篇文章主要讲解了“怎么使用不同的React hooks来解决日常所遇到的问题”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“怎么使用不同的React hoo...
    99+
    2024-04-02
  • mybatisplus的坑怎么解决
    这篇文章主要讲解了“mybatisplus的坑怎么解决”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“mybatisplus的坑怎么解决”吧!mybatisplus的坑 insert标签ins...
    99+
    2023-06-21
  • 怎么解决Static坑
    这篇文章主要介绍“怎么解决Static坑”,在日常操作中,相信很多人在怎么解决Static坑问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”怎么解决Static坑”的疑惑有所帮...
    99+
    2024-04-02
  • Golang的strings.Split()坑怎么解决
    这篇文章主要介绍“Golang的strings.Split()坑怎么解决”,在日常操作中,相信很多人在Golang的strings.Split()坑怎么解决问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”Gol...
    99+
    2023-06-30
  • python的ImageTk.PhotoImage坑怎么解决
    这篇文章主要介绍“python的ImageTk.PhotoImage坑怎么解决”,在日常操作中,相信很多人在python的ImageTk.PhotoImage坑怎么解决问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家...
    99+
    2023-07-04
  • mybatis的test坑怎么解决
    今天小编给大家分享一下mybatis的test坑怎么解决的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。概述mybatis使用...
    99+
    2023-07-05
  • react项目引入antd框架方式及遇到的坑怎么解决
    这篇文章主要介绍了react项目引入antd框架方式及遇到的坑怎么解决的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇react项目引入antd框架方式及遇到的坑怎么解决文章都会有所收获,下面我们一起来看看吧。r...
    99+
    2023-07-05
  • antd-react使用Select组件defaultValue踩的坑及解决
    目录需求背景问题描述原因分析前提当时想到的原因如下解决方案总结需求背景 有一个表格,点击表格每一行的一个叫"查看数据记录"的按钮,打开一个抽屉,会调后端一个接口(...
    99+
    2023-05-20
    antd-react使用Select组件 Select组件defaultValue踩坑 antd-react Select defaultValue踩坑
  • 详解JavaScheduledThreadPoolExecutor的踩坑与解决方法
    目录概述还原"大坑"解决方案更推荐的做法原理探究总结概述 最近项目上反馈某个重要的定时任务突然不执行了,很头疼,开发环境和测试环境都没有出现过这个问题。定时任务采...
    99+
    2022-11-13
    Java ScheduledThreadPoolExecutor
  • 使用aot的坑怎么解决
    本篇内容主要讲解“使用aot的坑怎么解决”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“使用aot的坑怎么解决”吧!启动参数固定化jaotc可以通过加-J参数来指定jvm的启动参数。我们尝试使用c...
    99+
    2023-06-03
  • python中os.path.exits()的坑怎么解决
    问题有时候我们经常会看到类似的代码:assert os.path.isdir(path_model_dir) assert os.path.exits(path_model_dir)如果填写一个相对路径作为path_model_dir,那将...
    99+
    2023-05-16
    Python
  • MySQL索引的坑怎么解决
    今天小编给大家分享一下MySQL索引的坑怎么解决的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下...
    99+
    2024-04-02
  • thinkphp6安装的坑怎么解决
    这篇文章主要介绍“thinkphp6安装的坑怎么解决”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“thinkphp6安装的坑怎么解决”文章能帮助大家解决问题。大坑==phpstudy 网站那 要关了...
    99+
    2023-06-30
  • js中setInterval多次调用问题怎么解决
    在 JavaScript 中,setInterval 函数可以用来重复执行一个函数或者一段代码。然而,有时候我们可能需要在 setI...
    99+
    2024-02-29
    js
  • ParallelStream使用的坑怎么解决
    今天小编给大家分享一下ParallelStream使用的坑怎么解决的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。比如下面的代...
    99+
    2023-06-17
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作