javascript - React functional component reinitialises local functions and variables on re render (React Hooks) - Stack Overflow

admin2025-04-03  1

So I have started using React hooks now. I have been experimenting with the API for some time now. i really like the idea of bringing the state to functional ponents. but there is this one thing which keeps on bothering me and it doesn't feel right in the gut when i am trying to use it. I tried posting on RFCs but it's too crowded there now. everything seems lost there.

Here is a piece of code from my example.

import React, { useState } from "react";

function Counter() {
  const [counterState,incrementCounterState] =  useCommontState(0); 

  function doSomething (){
   // does something and then calls incrementCounterState
   // with the updated state.
  }

   return (
    <div>
      <p>{counterState}</p>
      <button onClick={incrementCounterState}>increase</button>
      ....
      .... // some jsx calling local scoped functions.
      ....
    </div>
    );
}

function useCommontState(defaultValue){
  var [state, setState] = useState(0);
  function increment(){
    setState(defaultValue+=1);
  }
  return [state, increment]
}

export default Counter;

I can easily take out state and setState methods out and create a custom hook but my problem is with the local functions that are used by the ponent. since state is now part of the ponent there will be cases where some logic will decide what to do next with the state.

Also, when the ponent re-renders on state change everything gets reinitialized. which is my problem. I know that useState has its own way of handling the issue. but my problem is with my own functions. the click handlers. on change events, callbacks for child ponents etc. all that will be reinitialized everytime the ponent renders. this doesn't feel right to me.

Are there any ways by which we can work around it. it's a new API. we are not even sure if it will make into react 17. but has anyone e across any better way to do it?

So I have started using React hooks now. I have been experimenting with the API for some time now. i really like the idea of bringing the state to functional ponents. but there is this one thing which keeps on bothering me and it doesn't feel right in the gut when i am trying to use it. I tried posting on RFCs but it's too crowded there now. everything seems lost there.

Here is a piece of code from my example.

import React, { useState } from "react";

function Counter() {
  const [counterState,incrementCounterState] =  useCommontState(0); 

  function doSomething (){
   // does something and then calls incrementCounterState
   // with the updated state.
  }

   return (
    <div>
      <p>{counterState}</p>
      <button onClick={incrementCounterState}>increase</button>
      ....
      .... // some jsx calling local scoped functions.
      ....
    </div>
    );
}

function useCommontState(defaultValue){
  var [state, setState] = useState(0);
  function increment(){
    setState(defaultValue+=1);
  }
  return [state, increment]
}

export default Counter;

I can easily take out state and setState methods out and create a custom hook but my problem is with the local functions that are used by the ponent. since state is now part of the ponent there will be cases where some logic will decide what to do next with the state.

Also, when the ponent re-renders on state change everything gets reinitialized. which is my problem. I know that useState has its own way of handling the issue. but my problem is with my own functions. the click handlers. on change events, callbacks for child ponents etc. all that will be reinitialized everytime the ponent renders. this doesn't feel right to me.

Are there any ways by which we can work around it. it's a new API. we are not even sure if it will make into react 17. but has anyone e across any better way to do it?

Share Improve this question edited Nov 8, 2018 at 18:27 Yangshun Tay 53.3k33 gold badges123 silver badges150 bronze badges asked Nov 6, 2018 at 17:02 hannad rehmanhannad rehman 4,3615 gold badges37 silver badges59 bronze badges 2
  • not sure I understand your concern but I had a play earlier and ended up with useEffect(() => doSomething(), [false]) to do init work once, kind of of ponentDidMount - see reactjs/docs/… – Dimitar Christoff Commented Nov 6, 2018 at 17:30
  • I guess what react people believe is that the cost of functional re-initialization is quite insignificant than the other problems that it solves. – Shubham Khatri Commented Nov 6, 2018 at 17:52
Add a ment  | 

3 Answers 3

Reset to default 7

I had the same concerns as well when I first saw the proposal, but this was addressed in the React Docs Hooks Proposal FAQ:

Are Hooks slow because of creating functions in render?

No. In modern browsers, the raw performance of closures pared to classes doesn’t differ significantly except in extreme scenarios.

My takeaway is that although you have additional overhead now in the repeated declarations per render, you have additional wins elsewhere:

  • Hooks avoid a lot of the overhead that classes require, like the cost of creating class instances and binding event handlers in the constructor.

  • Idiomatic code using Hooks doesn’t need the deep ponent tree nesting that is prevalent in codebases that use higher-order ponents, render props, and context. With smaller ponent trees, React has less work to do.

Overall the benefits might be more than the downsides which makes hooks worth using.

You can always simplify the code to take functions out so that they aren't initialised always, by passing the required values as constants.

    import React, { useState } from "react";

    function doSomething (counterState, incrementCounterState){
       // does something and then calls incrementCounterState
       // with the updated state.
   }
    function Counter() {
      const [counterState,incrementCounterState] =  useCommontState(0); 

       return (
        <div>
          <p>{counterState}</p>
          <button onClick={incrementCounterState}>increase</button>
          ....
          .... // some jsx calling local scoped functions.
          ....
        </div>
        );
    }

    function increment(defaultValue, setState){
        setState(defaultValue + 1);
    }

    function useCommontState(defaultValue){
      var [state, setState] = useState(0);
      return [state, increment]
    }

    export default Counter;

Also in my opinion the function design being suggested in all the demos and docs is for people to get fortable with it and then think about the re-initialization aspects. Also the cost that re-initialization would significanly be overpowered by the other benefits that it provides.

I'm using createOnce helper function to prevent reinitialises, But I'm not sure if it's correct or not.

utils/createOnce.js

import { useMemo } from 'react';

export const createOnce = toCreate => useMemo(() => toCreate, []);

SomeComponent.js

...

const someFunction = createOnce((counter) => {
  // whatever
  return counter + 1;
});

...
转载请注明原文地址:http://conceptsofalgorithm.com/Algorithm/1743630365a213896.html

最新回复(0)