Согласно документам:
componentDidUpdate()
вызывается сразу после обновления. Этот метод не вызывается для первоначального рендеринга.
Мы можем использовать новый useEffect()
хук для моделирования componentDidUpdate()
, но похоже, что useEffect()
он запускается после каждого рендеринга, даже в первый раз. Как мне заставить его не запускаться при начальном рендере?
Как вы можете видеть в приведенном ниже примере, componentDidUpdateFunction
он печатается во время первоначального рендеринга, но componentDidUpdateClass
не печатается во время первоначального рендеринга.
function ComponentDidUpdateFunction() {
const [count, setCount] = React.useState(0);
React.useEffect(() => {
console.log("componentDidUpdateFunction");
});
return (
<div>
<p>componentDidUpdateFunction: {count} times</p>
<button
onClick={() => {
setCount(count + 1);
}}
>
Click Me
</button>
</div>
);
}
class ComponentDidUpdateClass extends React.Component {
constructor(props) {
super(props);
this.state = {
count: 0,
};
}
componentDidUpdate() {
console.log("componentDidUpdateClass");
}
render() {
return (
<div>
<p>componentDidUpdateClass: {this.state.count} times</p>
<button
onClick={() => {
this.setState({ count: this.state.count + 1 });
}}
>
Click Me
</button>
</div>
);
}
}
ReactDOM.render(
<div>
<ComponentDidUpdateFunction />
<ComponentDidUpdateClass />
</div>,
document.querySelector("#app")
);
<script src="https://unpkg.com/react@16.7.0-alpha.0/umd/react.development.js"></script>
<script src="https://unpkg.com/react-dom@16.7.0-alpha.0/umd/react-dom.development.js"></script>
<div id="app"></div>
javascript
reactjs
react-hooks
Яншун Тай
источник
источник
count
?Ответы:
Мы можем использовать
useRef
ловушку для хранения любого изменяемого значения, которое нам нравится , поэтому мы можем использовать это, чтобы отслеживатьuseEffect
, запускается ли функция в первый раз .Если мы хотим, чтобы эффект выполнялся в той же фазе, что и
componentDidUpdate
он, мы можем использоватьuseLayoutEffect
вместо этого.пример
const { useState, useRef, useLayoutEffect } = React; function ComponentDidUpdateFunction() { const [count, setCount] = useState(0); const firstUpdate = useRef(true); useLayoutEffect(() => { if (firstUpdate.current) { firstUpdate.current = false; return; } console.log("componentDidUpdateFunction"); }); return ( <div> <p>componentDidUpdateFunction: {count} times</p> <button onClick={() => { setCount(count + 1); }} > Click Me </button> </div> ); } ReactDOM.render( <ComponentDidUpdateFunction />, document.getElementById("app") );
<script src="https://unpkg.com/react@16.7.0-alpha.0/umd/react.development.js"></script> <script src="https://unpkg.com/react-dom@16.7.0-alpha.0/umd/react-dom.development.js"></script> <div id="app"></div>
источник
useRef
наuseState
, но использование сеттера вызвало повторный рендеринг, чего не происходит при назначении,firstUpdate.current
поэтому я думаю, что это единственный хороший способ :)componentDidUpdate
с помощью хуков, поэтому я использовал его.Вы можете превратить его в пользовательские хуки , например:
import React, { useEffect, useRef } from 'react'; const useDidMountEffect = (func, deps) => { const didMount = useRef(false); useEffect(() => { if (didMount.current) func(); else didMount.current = true; }, deps); } export default useDidMountEffect;
Пример использования:
import React, { useState, useEffect } from 'react'; import useDidMountEffect from '../path/to/useDidMountEffect'; const MyComponent = (props) => { const [state, setState] = useState({ key: false }); useEffect(() => { // you know what is this, don't you? }, []); useDidMountEffect(() => { // react please run me if 'key' changes, but not on initial render }, [state.key]); return ( <div> ... </div> ); } // ...
источник
Я сделал простой
useFirstRender
крючок для обработки таких случаев, как фокусирование ввода формы:import { useRef, useEffect } from 'react'; export function useFirstRender() { const firstRender = useRef(true); useEffect(() => { firstRender.current = false; }, []); return firstRender.current; }
Он начинается как
true
, затем переключается наfalse
вuseEffect
, который запускается только один раз и больше никогда.В своем компоненте используйте его:
const firstRender = useFirstRender(); const phoneNumberRef = useRef(null); useEffect(() => { if (firstRender || errors.phoneNumber) { phoneNumberRef.current.focus(); } }, [firstRender, errors.phoneNumber]);
В вашем случае вы просто используете
if (!firstRender) { ...
.источник
@ravi, ваш не вызывает переданную функцию размонтирования. Вот версия, которая немного более полная:
/** * Identical to React.useEffect, except that it never runs on mount. This is * the equivalent of the componentDidUpdate lifecycle function. * * @param {function:function} effect - A useEffect effect. * @param {array} [dependencies] - useEffect dependency list. */ export const useEffectExceptOnMount = (effect, dependencies) => { const mounted = React.useRef(false); React.useEffect(() => { if (mounted.current) { const unmount = effect(); return () => unmount && unmount(); } else { mounted.current = true; } }, dependencies); // Reset on unmount for the next mount. React.useEffect(() => { return () => mounted.current = false; }, []); };
источник
useEffect(() => {...});
dependencies
параметр при его вызове.@MehdiDehghani, ваше решение работает отлично, вам нужно сделать одно дополнение, это отключить, сбросить
didMount.current
значение наfalse
. Когда пытаться использовать этот настраиваемый хук где-нибудь еще, вы не получите значение кеша.import React, { useEffect, useRef } from 'react'; const useDidMountEffect = (func, deps) => { const didMount = useRef(false); useEffect(() => { let unmount; if (didMount.current) unmount = func(); else didMount.current = true; return () => { didMount.current = false; unmount && unmount(); } }, deps); } export default useDidMountEffect;
источник
false
.