191

Let me explain the result of this code for asking my issue easily.

const ForExample = () => {
    const [name, setName] = useState('');
    const [username, setUsername] = useState('');

    useEffect(() => {
        console.log('effect');
        console.log({
            name,
            username
        });

        return () => {
            console.log('cleaned up');
            console.log({
                name,
                username
            });
        };
    }, [username]);

    const handleName = e => {
        const { value } = e.target;

        setName(value);
    };

    const handleUsername = e => {
        const { value } = e.target;

        setUsername(value);
    };

    return (
        <div>
            <div>
                <input value={name} onChange={handleName} />
                <input value={username} onChange={handleUsername} />
            </div>
            <div>
                <div>
                    <span>{name}</span>
                </div>
                <div>
                    <span>{username}</span>
                </div>
            </div>
        </div>
    );
};

When the ForExample component mounts, 'effect' will be logged. This is related to the componentDidMount().

And whenever I change name input, both 'effect' and 'cleaned up' will be logged. Vice versa, no message will be logged whenever I change username input since I added [username] to the second parameter of useEffect(). This is related to the componentDidUpdate()

Lastly, when the ForExample component unmounts, 'cleaned up' will be logged. This is related to the componentWillUnmount().

We all know that.

To sum, 'cleaned up' is invoked whenever the component is being re-rendered(includes unmount)

If I want to make this component to log 'cleaned up' for only the moment when it is unmount, I just have to change the second parameter of useEffect() to [].

But If I change [username] to [], ForExample component no longer implements the componentDidUpdate() for name input.

What I want to do is that, to make the component supports both componentDidUpdate() only for name input and componentWillUnmount(). (logging 'cleaned up' for only the moment when the component is being unmounted)

koo
  • 4,013
  • 6
  • 15
  • 31
  • 8
    You could have 2 separate effects. One which is given an array with `username` in it as second argument, and one that is given an empty array as second argument. – Tholle Mar 06 '19 at 10:02
  • @Tholle Do you mean I have to make 2 seperate useEffect() methods? – koo Mar 06 '19 at 10:05
  • 1
    Yes, that's one way to go about it. – Tholle Mar 06 '19 at 10:06
  • 1
    @Tholle I thought it would be overrided by the last useEffect() method. I'll try. Thanks – koo Mar 06 '19 at 10:07
  • @Tholle It works. thanks again. By the way, is there any prettier way to implement this? It feels like we write same-named methods twice. – koo Mar 06 '19 at 10:11
  • 2
    Great! You're welcome. It depends on what the cleanup should do. 2 separate effects is not a bad solution. – Tholle Mar 06 '19 at 10:12
  • @Tholle for me not works, i have work with redux, and i have a method which do flush all dat when component is unloaded, but not working for me, Help...... – NsdHSO Aug 07 '21 at 08:30
  • @Racal Post your question, elaborate your issue with some codes, call me out with the tag, and maybe I can help you. – koo Aug 19 '21 at 08:55
  • @koo i resolved this issue, but we can make friend on github. Thank you. – NsdHSO Aug 21 '21 at 16:27

12 Answers12

261

You can use more than one useEffect().

For example, if my variable is data1, I can use all of this in my component:

useEffect( () => console.log("mount"), [] );
useEffect( () => console.log("data1 update"), [ data1 ] );
useEffect( () => console.log("any update") );
useEffect( () => () => console.log("data1 update or unmount"), [ data1 ] );
useEffect( () => () => console.log("unmount"), [] );
Cormor
  • 59
  • 7
Mehran Motiee
  • 3,547
  • 1
  • 13
  • 16
  • 13
    what is the difference between first and last useEffects, first useEffect will be invoked on the **willmount** or **didmount**, last useEffect was returned call back function with empty array why? could you elaborate each useEffect use cases when and how we can use? – Siluveru Kiran Kumar Aug 02 '19 at 10:46
  • 8
    @siluverukirankumar Return value (function) of a callback is what is being called on destroy (unmount event). That is why the last example is a HOC, returning function immediately. The second parameter is where React would look for changes to rerun this hook. When it is an empty array it would run just once. – Georgy Aug 27 '19 at 22:06
  • 4
    Thanks @Georgy got it the last useEffect is returning callback not seen clearly – Siluveru Kiran Kumar Aug 28 '19 at 05:26
  • 4
    So, if you make a useEffect hook, and it returns a function .. then the code before the returned function runs as a componentDidMount ... and the code in the returned function gets called for componentWillUnmount? It's a little confusing, so making sure I understand right. useEffect(()=>{ // code to run on mount ... return()=> { //code to run dismount}}) Is that right? – Maiya Nov 18 '19 at 16:43
  • 3
    I suggest reading https://overreacted.io/a-complete-guide-to-useeffect/ Thinking about hooks in lifecycles isn't really that sweet – moto Nov 22 '19 at 01:13
  • This is nice. But I believe your second and third useEffect should read "did update" rather than "will update". – Jordy May 07 '20 at 05:17
156

Since the cleanup is not dependent on the username, you could put the cleanup in a separate useEffect that is given an empty array as second argument.

Example

const { useState, useEffect } = React;

const ForExample = () => {
  const [name, setName] = useState("");
  const [username, setUsername] = useState("");

  useEffect(
    () => {
      console.log("effect");
    },
    [username]
  );

  useEffect(() => {
    return () => {
      console.log("cleaned up");
    };
  }, []);

  const handleName = e => {
    const { value } = e.target;

    setName(value);
  };

  const handleUsername = e => {
    const { value } = e.target;

    setUsername(value);
  };

  return (
    <div>
      <div>
        <input value={name} onChange={handleName} />
        <input value={username} onChange={handleUsername} />
      </div>
      <div>
        <div>
          <span>{name}</span>
        </div>
        <div>
          <span>{username}</span>
        </div>
      </div>
    </div>
  );
};

function App() {
  const [shouldRender, setShouldRender] = useState(true);

  useEffect(() => {
    setTimeout(() => {
      setShouldRender(false);
    }, 5000);
  }, []);

  return shouldRender ? <ForExample /> : null;
}

ReactDOM.render(<App />, document.getElementById("root"));
<script src="https://unpkg.com/react@16/umd/react.development.js"></script>
<script src="https://unpkg.com/react-dom@16/umd/react-dom.development.js"></script>

<div id="root"></div>
Tholle
  • 108,070
  • 19
  • 198
  • 189
  • 1
    nice and clean example. I am wondering though. Can I trigger a use effect somehow on changed navigation, or will I have to move it up in the components tree? Because when just pasting your "cleaned up" useEffect I am not seeing this trigger. – Fabian Bosler Apr 22 '20 at 11:06
32

To add to the accepted answer, I had a similar issue and solved it using a similar approach with the contrived example below. In this case I needed to log some parameters on componentWillUnmount and as described in the original question I didn't want it to log every time the params changed.

const componentWillUnmount = useRef(false)

// This is componentWillUnmount
useEffect(() => {
    return () => {
        componentWillUnmount.current = true
    }
}, [])

useEffect(() => {
    return () => {
        // This line only evaluates to true after the componentWillUnmount happens 
        if (componentWillUnmount.current) {
            console.log(params)
        }
    }

}, [params]) // This dependency guarantees that when the componentWillUnmount fires it will log the latest params
Barry
  • 722
  • 1
  • 9
  • 13
10

You can simply write it as :

  useEffect(() => {
  return () => {};
  }, []);
Manik Kumar
  • 764
  • 7
  • 17
4

Using custom js events you can emulate unmounting a componentWillUnmount even when having dependency.

Problem:

    useEffect(() => {
    //Dependent Code
    return () => {
        // Desired to perform action on unmount only 'componentWillUnmount' 
        // But it does not
        if(somethingChanged){
            // Perform an Action only if something changed
        }
    }
},[somethingChanged]);

Solution:

// Rewrite this code  to arrange emulate this behaviour

// Decoupling using events
useEffect( () => {
    return () => {
        // Executed only when component unmounts,
        let e = new Event("componentUnmount");
        document.dispatchEvent(e);
    }
}, []);

useEffect( () => {
    function doOnUnmount(){
        if(somethingChanged){
            // Perform an Action only if something changed
        }
    }

    document.addEventListener("componentUnmount",doOnUnmount);
    return () => {
        // This is done whenever value of somethingChanged changes
        document.removeEventListener("componentUnmount",doOnUnmount);
    }

}, [somethingChanged])

Caveats: useEffects have to be in order, useEffect with no dependency have to be written before, this is to avoid the event being called after its removed.

SS7
  • 84
  • 3
  • 1
    The most important part of this answer is the caveat that you have to use useEffect hooks in order – brycejl Jul 15 '21 at 15:40
3
function LegoComponent() {

  const [lego, setLegos] = React.useState([])

  React.useEffect(() => {
    let isSubscribed = true
    fetchLegos().then( legos=> {
      if (isSubscribed) {
        setLegos(legos)
      }
    })
    return () => isSubscribed = false
  }, []);

  return (
    <ul>
    {legos.map(lego=> <li>{lego}</li>)}
    </ul>
  )
}

In the code above, the fetchLegos function returns a promise. We can “cancel” the promise by having a conditional in the scope of useEffect, preventing the app from setting state after the component has unmounted.

Warning: Can't perform a React state update on an unmounted component. This is a no-op, but it indicates a memory leak in your application. To fix, cancel all subscriptions and asynchronous tasks in a useEffect cleanup function.

tek foo
  • 59
  • 1
  • 7
2

instead of creating too many complicated functions and methods what I do is I create an event listener and automatically have mount and unmount done for me without having to worry about doing it manually. Here is an example.

//componentDidMount
useEffect( () => {

    window.addEventListener("load",  pageLoad);

    //component will unmount
    return () => {
       
        window.removeEventListener("load", pageLoad);
    }

 });

now that this part is done I just run anything I want from the pageLoad function like this.

const pageLoad = () =>{
console.log(I was mounted and unmounted automatically :D)}
jerryurenaa
  • 3,863
  • 1
  • 27
  • 17
2

Here is my solution, generalized into a custom hook:

import React, { useEffect, useRef } from 'react';

const useUnmountEffect = (effect, dependencies) => {
  if (typeof effect !== 'function') {
    console.error('Effect must be a function');
  }

  const componentWillUnmount = useRef(false)

  useEffect(() => () => {
    componentWillUnmount.current = true
  }, []);

  useEffect(() => () => {
    if (componentWillUnmount.current) {
      effect?.();
    }
  }, dependencies);
}

export default useUnmountEffect;
Brad Stiff
  • 69
  • 5
1

what about:

function useOnUnmount(callback: () => void) {
    const onUnmount = useRef<(() => void) | null>(null);
    onUnmount.current = callback;

    useEffect(() => {
        return () => onUnmount.current?.();
    }, []);
}

useOnUnmount(() => {
    console.log("unmount", props);
});
Michael Knoch
  • 151
  • 2
  • 5
0

useEffect are isolated within its own scope and gets rendered accordingly. Image from https://reactjs.org/docs/hooks-custom.html

enter image description here

ksav
  • 20,015
  • 6
  • 46
  • 66
  • A link to a solution is welcome, but please ensure your answer is useful without it: [add context around the link](//meta.stackexchange.com/a/8259) so your fellow users will have some idea what it is and why it’s there, then quote the most relevant part of the page you're linking to in case the target page is unavailable. [Answers that are little more than a link may be deleted](//stackoverflow.com/help/deleted-answers). – rizerphe May 10 '20 at 05:22
0

Using two useEffect cleanups as seen in other solutions is not guaranteed because React does not guarantee the cleanup order of sibling effects.

You could try something like this, which will fire on component unmount or window close if enabled, whichever comes first. The first parameter is the callback, which you can change during the component's lifetime. The 2nd parameter is whether to enable the window closing event listener, which may only be set initially to avoid a scenario where we might have to rapidly register and unregister the same event listener, which can fail.

Note, if you see this firing once on mount and again on unmount, it is probably because you are in development mode and have React.StrictMode turned on. At least as of React 18, this will mount your component, unmount it, and then mount it again.

import { useEffect, useState } from "react";

export default function useWillUnmount(callback, beforeUnload = false) {
  const [cache] = useState({beforeUnload});
  cache.call = callback;

  useEffect(() => {
    let cancel = false;

    function unmount() {
      if (cancel) return;
      cancel = true;
      cache.call?.();
    }

    if (cache.beforeUnload) window.addEventListener("beforeunload", unmount);

    return () => {
      if (cache.beforeUnload)
        window.removeEventListener("beforeunload", unmount);
      unmount();
    }
  }, [cache]); // this makes the effect run only once per component lifetime
}
Chris Chiasson
  • 547
  • 8
  • 17
0

use ref to add the callback and use wisely.

import { useEffect, useRef } from "react";
function useUnMount(run: () => void) {
  const referencedRun = useRef(run);
  useEffect(
    () => () => {
      if (referencedRun.current) {
        referencedRun.current();
      }
    },
    []
  );
}

export default useUnMount;

and use it like

  useUnMount(() => {
    console.log("called on unmount");
  });

with React 18 or Strict mode

import { useEffect, useRef } from "react";
export const isProd = process.env.NODE_ENV == "production";
export const isDev = process.env.NODE_ENV == "development";

function useUnMount(run: () => void) {
  const referencedRun = useRef(run);
  const times = useRef(0);
  useEffect(
    () => () => {
      times.current += 1;
      const call = isProd || (isDev && times.current == 2);
      if (call) {
        referencedRun.current();
      }
    },
    []
  );
}

export default useUnMount;
Mukundhan
  • 3,284
  • 23
  • 36