微光掠影 發表於 2020-10-17 15:14:00

React Hooks与setInterval

<h3>前言</h3>
<p>Hooks出来已经有段时间了,相信大家都用过段时间了,有没有小伙伴们遇到坑呢,我这边就有个 setInterval&nbsp;的坑,和小伙伴们分享下解决方案。</p>
<p>&nbsp;</p>
<h3>前言</h3>
<p>写个 count&nbsp;每秒自增的定时器,如下写法结果,界面上 count&nbsp;为 1&nbsp;?</p>
<pre><code class="hljs javascript"><span class="hljs-function"><span class="hljs-keyword">function <span class="hljs-title">Counter(<span class="hljs-params">) {
<span class="hljs-keyword">let = useState(<span class="hljs-number">0);
useEffect(<span class="hljs-function"><span class="hljs-params">() =&gt; {
    <span class="hljs-keyword">let id = setInterval(<span class="hljs-function"><span class="hljs-params">() =&gt; {
      setCount(count + <span class="hljs-number">1);
    }, <span class="hljs-number">1000);
    <span class="hljs-keyword">return <span class="hljs-function"><span class="hljs-params">() =&gt; clearInterval(id);
}, []);
<span class="hljs-keyword">return <span class="xml"><span class="hljs-tag">&lt;<span class="hljs-name">h1&gt;{count}<span class="hljs-tag">&lt;/<span class="hljs-name">h1&gt;;
}</span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></code></pre>
<p>https://codesandbox.io/embed/hooks-setinterval-error-w4qu6</p>
<p>如果某些特定值在两次重渲染之间没有发生变化,你可以通知&nbsp;react&nbsp;跳过对 effect&nbsp;的调用。就是将第二个参数改成 []&nbsp;,类似于更接近类组件的 componentDidMount&nbsp;和 componentWillUnmount&nbsp;生命周期,只执行一次。 effect&nbsp;的第二个参数中传入的值就是 它更改的话, effect&nbsp;也会重新执行一遍的值。</p>
<p>因为 Effect&nbsp;的第二个参数为 []&nbsp;,没有依赖, Effect&nbsp;只会执行一次。 setInterval&nbsp;中拿到的count&nbsp;永远是 0&nbsp;,界面会一直显示 1&nbsp;,如下所示:</p>
<pre><code class="hljs javascript"><span class="hljs-function"><span class="hljs-keyword">function <span class="hljs-title">Counter(<span class="hljs-params">) {
<span class="hljs-keyword">let = useState(<span class="hljs-number">0);
useEffect(<span class="hljs-function"><span class="hljs-params">() =&gt; {
    <span class="hljs-keyword">let id = setInterval(<span class="hljs-function"><span class="hljs-params">() =&gt; {
      setCount(<span class="hljs-number">0 + <span class="hljs-number">1);
    }, <span class="hljs-number">1000);
    <span class="hljs-keyword">return <span class="hljs-function"><span class="hljs-params">() =&gt; clearInterval(id);
}, []);
<span class="hljs-keyword">return <span class="xml"><span class="hljs-tag">&lt;<span class="hljs-name">h1&gt;{count}<span class="hljs-tag">&lt;/<span class="hljs-name">h1&gt;;
}</span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></code></pre>
<p>那有些小伙伴会说,如果我们直接往第二个参数加 count&nbsp;呢</p>
<pre><code class="hljs javascript"><span class="hljs-function"><span class="hljs-keyword">function <span class="hljs-title">Counter(<span class="hljs-params">) {
<span class="hljs-comment">//...
useEffect(<span class="hljs-function"><span class="hljs-params">() =&gt; {
    <span class="hljs-keyword">let id = setInterval(<span class="hljs-function"><span class="hljs-params">() =&gt; {
      setCount(count + <span class="hljs-number">1);
    }, <span class="hljs-number">1000);
    <span class="hljs-keyword">return <span class="hljs-function"><span class="hljs-params">() =&gt; clearInterval(id);
}, );
<span class="hljs-comment">//...
}</span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></code></pre>
<p>这样效果是对的,但是性能不好。每当 count&nbsp;更改了, useEffect&nbsp;就会渲染一次,定时器也会不停的被新增与移除。如下所示:</p>
<pre><code class="hljs javascript"><span class="hljs-comment">//第一次
<span class="hljs-function"><span class="hljs-keyword">function <span class="hljs-title">Counter(<span class="hljs-params">) {
<span class="hljs-comment">//...
useEffect(<span class="hljs-function"><span class="hljs-params">() =&gt; {
    <span class="hljs-keyword">let id = setInterval(<span class="hljs-function"><span class="hljs-params">() =&gt; {
      setCount(<span class="hljs-number">0 + <span class="hljs-number">1);
    }, <span class="hljs-number">1000);
    <span class="hljs-keyword">return <span class="hljs-function"><span class="hljs-params">() =&gt; clearInterval(id);
}, [<span class="hljs-number">0]);
<span class="hljs-comment">//...
}
<span class="hljs-comment">//第二次
<span class="hljs-function"><span class="hljs-keyword">function <span class="hljs-title">Counter(<span class="hljs-params">) {
<span class="hljs-comment">//...
useEffect(<span class="hljs-function"><span class="hljs-params">() =&gt; {
    <span class="hljs-keyword">let id = setInterval(<span class="hljs-function"><span class="hljs-params">() =&gt; {
      setCount(<span class="hljs-number">1 + <span class="hljs-number">1);
    }, <span class="hljs-number">1000);
    <span class="hljs-keyword">return <span class="hljs-function"><span class="hljs-params">() =&gt; clearInterval(id);
}, [<span class="hljs-number">1]);
<span class="hljs-comment">//...
<span class="hljs-comment">//第N次
}</span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></code></pre>
<p>那到底要怎么做才能有保障性能,定时器只监听一次,又使定时器起作用呢?</p>
<p>&nbsp;</p>
<h3>方案一、函数式更新</h3>
<p>useState&nbsp;中的set方法可接收函数,该函数将接收先前的 state&nbsp;,并返回一个更新后的值。这样定时器每次拿到的是最新的值。</p>
<pre><code class="hljs javascript"><span class="hljs-function"><span class="hljs-keyword">function <span class="hljs-title">Counter(<span class="hljs-params">) {
<span class="hljs-keyword">let = useState(<span class="hljs-number">0);
useEffect(<span class="hljs-function"><span class="hljs-params">() =&gt; {
    <span class="hljs-keyword">let id = setInterval(<span class="hljs-function"><span class="hljs-params">() =&gt; {
      setCount(<span class="hljs-function"><span class="hljs-params">v =&gt; {
      <span class="hljs-keyword">return v + <span class="hljs-number">1;
      });
    }, <span class="hljs-number">1000);
    <span class="hljs-keyword">return <span class="hljs-function"><span class="hljs-params">() =&gt; clearInterval(id);
}, []);
<span class="hljs-keyword">return <span class="xml"><span class="hljs-tag">&lt;<span class="hljs-name">h1&gt;{count}<span class="hljs-tag">&lt;/<span class="hljs-name">h1&gt;;
}</span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></code></pre>
<p>https://codesandbox.io/embed/hooks-setinterval-usestate-grres</p>
<p>&nbsp;</p>
<h3>方案二、用useRef</h3>
<p>useRef&nbsp;返回一个可变的 ref&nbsp;对象,返回的 ref&nbsp;对象在组件的整个生命周期内保持不变。</p>
<p>将定时器函数提取出来,每次定时器触发时,都能取到最新到 count&nbsp;.</p>
<pre><code class="hljs javascript"><span class="hljs-function"><span class="hljs-keyword">function <span class="hljs-title">Counter(<span class="hljs-params">) {
<span class="hljs-keyword">let = useState(<span class="hljs-number">0);
<span class="hljs-keyword">const</span> myRef = useRef(<span class="hljs-literal">null);
myRef.current = <span class="hljs-function"><span class="hljs-params">() =&gt; {
    setCount(count + <span class="hljs-number">1);
};
useEffect(<span class="hljs-function"><span class="hljs-params">() =&gt; {
    <span class="hljs-keyword">let id = setInterval(<span class="hljs-function"><span class="hljs-params">() =&gt; {
      myRef.current();
    }, <span class="hljs-number">1000);
    <span class="hljs-keyword">return <span class="hljs-function"><span class="hljs-params">() =&gt; clearInterval(id);
}, []);
<span class="hljs-keyword">return <span class="xml"><span class="hljs-tag">&lt;<span class="hljs-name">h1&gt;{count}<span class="hljs-tag">&lt;/<span class="hljs-name">h1&gt;;
}</span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></code></pre>
<p>https://codesandbox.io/embed/hooks-setinterval-useref-cgif3</p>
<p>思考:为什么不直接 setInterval(myRef.current, 1000)&nbsp;这样写不行呢,还要包个方法返回?</p>
<pre><code class="hljs javascript"><span class="hljs-function"><span class="hljs-keyword">function <span class="hljs-title">Counter(<span class="hljs-params">) {
<span class="hljs-keyword">let = useState(<span class="hljs-number">0);
<span class="hljs-keyword">const</span> myRef = useRef(<span class="hljs-literal">null);
myRef.current = <span class="hljs-function"><span class="hljs-params">() =&gt; {
    setCount(count + <span class="hljs-number">1);
};
useEffect(<span class="hljs-function"><span class="hljs-params">() =&gt; {
    <span class="hljs-keyword">let id = setInterval(myRef.current, <span class="hljs-number">1000);
    <span class="hljs-keyword">return <span class="hljs-function"><span class="hljs-params">() =&gt; clearInterval(id);
}, []);
<span class="hljs-keyword">return <span class="xml"><span class="hljs-tag">&lt;<span class="hljs-name">h1&gt;{count}<span class="hljs-tag">&lt;/<span class="hljs-name">h1&gt;;
}</span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></code></pre>
<p>https://codesandbox.io/embed/hooks-setinterval-useref-error-52dm0</p>
<p>下面的例子可以很好的解释。假如把 myRef.current&nbsp;为 cur&nbsp;变量,定时器的第一个参数为 interval&nbsp;变量, cur&nbsp;变量更改, interval&nbsp;的取的还是之前赋值的值。</p>
<pre><code class="hljs javascript"><span class="hljs-keyword">var</span> cur=<span class="hljs-function"><span class="hljs-params">()=&gt;{<span class="hljs-keyword">var</span> count=<span class="hljs-number">0;<span class="hljs-built_in">console.log(count)};
<span class="hljs-keyword">var interval=cur;
<span class="hljs-keyword">var cur=<span class="hljs-function"><span class="hljs-params">()=&gt;{<span class="hljs-keyword">var count=<span class="hljs-number">1;<span class="hljs-built_in">console.log(count)};
interval();<span class="hljs-comment">//0

<span class="hljs-keyword">var cur=<span class="hljs-function"><span class="hljs-params">()=&gt;{<span class="hljs-keyword">var count=<span class="hljs-number">0;<span class="hljs-built_in">console.log(count)};
<span class="hljs-keyword">var interval=<span class="hljs-function"><span class="hljs-params">()=&gt;{cur()};
<span class="hljs-keyword">var cur=<span class="hljs-function"><span class="hljs-params">()=&gt;{<span class="hljs-keyword">var count=<span class="hljs-number">1;<span class="hljs-built_in">console.log(count)};
interval();<span class="hljs-comment">//1</span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></code></pre>
<p>&nbsp;</p>
<h3>方案三、自定义hook</h3>
<p>可以写个自定义 hook&nbsp;,方便重复使用。</p>
<pre><code class="hljs javascript"><span class="hljs-function"><span class="hljs-keyword">function <span class="hljs-title">useInterval(<span class="hljs-params">fun) {
<span class="hljs-keyword">const myRef = useRef(<span class="hljs-literal">null);
useEffect(<span class="hljs-function"><span class="hljs-params">() =&gt; {
    myRef.current = fun;
}, );
useEffect(<span class="hljs-function"><span class="hljs-params">() =&gt; {
    <span class="hljs-keyword">let id = setInterval(<span class="hljs-function"><span class="hljs-params">() =&gt; {
      myRef.current();
    }, <span class="hljs-number">1000);
    <span class="hljs-keyword">return <span class="hljs-function"><span class="hljs-params">() =&gt; clearInterval(id);
}, []);
}

<span class="hljs-function"><span class="hljs-keyword">function <span class="hljs-title">Counter(<span class="hljs-params">) {
<span class="hljs-keyword">let = useState(<span class="hljs-number">0);
useInterval(<span class="hljs-function"><span class="hljs-params">() =&gt; {
    setCount(count + <span class="hljs-number">1);
});
<span class="hljs-keyword">return <span class="xml"><span class="hljs-tag">&lt;<span class="hljs-name">h1&gt;{count}<span class="hljs-tag">&lt;/<span class="hljs-name">h1&gt;;
}</span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></code></pre>
<p>https://codesandbox.io/embed/hooks-setinterval-ownhooks-0tpxe</p>
<p><span style="position: relative; left: -100000px">佛山vi设计https://www.houdianzi.com/fsvi/</span> <span style="position: relative; left: -100000px">豌豆资源搜索大全https://55wd.com</span></p>
<h3>方案四、用useReducer</h3>
<p>将 count&nbsp;变量存入 reducer&nbsp;中,使用 useReducer&nbsp;更新 count</p>
<pre><code class="hljs javascript"><span class="hljs-function"><span class="hljs-keyword">function <span class="hljs-title">reducer(<span class="hljs-params">state, action) {
<span class="hljs-keyword">switch (action.type) {
    <span class="hljs-keyword">case <span class="hljs-string">"increment":
      <span class="hljs-keyword">return state + <span class="hljs-number">1;
    <span class="hljs-keyword">default:
      <span class="hljs-keyword">throw <span class="hljs-keyword">new <span class="hljs-built_in">Error();
}
}

<span class="hljs-function"><span class="hljs-keyword">function <span class="hljs-title">Counter(<span class="hljs-params">) {
<span class="hljs-keyword">const = useReducer(reducer, <span class="hljs-number">0);
useEffect(<span class="hljs-function"><span class="hljs-params">() =&gt; {
    setInterval(<span class="hljs-function"><span class="hljs-params">() =&gt; {
      dispatch({ <span class="hljs-attr">type: <span class="hljs-string">"increment" });
    }, <span class="hljs-number">1000);
}, []);
<span class="hljs-keyword">return <span class="xml"><span class="hljs-tag">&lt;<span class="hljs-name">h1&gt;{state}<span class="hljs-tag">&lt;/<span class="hljs-name">h1&gt;;
}</span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></code></pre><br><br>
来源:https://www.cnblogs.com/xiaonian8/p/13831339.html
頁: [1]
查看完整版本: React Hooks与setInterval