3 Rules of React State Management

栏目: IT技术 · 发布时间: 4年前

内容简介:State inside a React component is the encapsulated data that is persistent between renderings.I like thatThis post describes 3 easy rules that answer the above questions and help you design the component’s state.

State inside a React component is the encapsulated data that is persistent between renderings. useState() is the React hook responsible for managing state inside a functional component.

I like that useState() indeed makes the work with state quite easy. But often I encounter questions like:

  • should I divide my component’s state into small states, or keep a compound one?
  • if the state management becomes complicated, should I extract it from the component? How to do that?
  • if useState() usage is so simple, when would you need useReducer() ?

This post describes 3 easy rules that answer the above questions and help you design the component’s state.

1. One concern

The first good rule of efficient state management is:

Make a state variable responsible for one concern.

Having a state variable responsible for one concern makes it conform to the Single Responsibility Principle.

Let’s see an example of a compound state, i.e. a state that incorporates multiple state values.

const [state, setState] = useState({
  on: true,
  count: 0
});

state.on    // => true
state.count // => 0

The state consists of a plain JavaScript object, having the properties on and count .

The first property, state.on , holds a boolean denoting a switch. The same way state.count holds a number denoting a counter, for example, how many times the user had clicked a button.

Then, let’s say you’d like to increase the counter by 1:

// Updating compound state
setUser({
  ...state,
  count: state.count + 1
});

You have to keep nearby the whole state to be able to update just count . This is a big construction to invoke to simply increase a counter: all because the state variable is responsible for 2 concerns: switch and counter.

The solution is to split the compound state into 2 atomic states on and count :

const [on, setOnOff] = useState(true);
const [count, setCount] = useState(0);

on state variable is solely responsible for storing the switch state. The same way count variable is solely responsible for a counter.

Now let’s try to update the counter:

setCount(count + 1);
// or using a callback
setCount(count => count + 1);

count state, which is responsible for counting only, is easy to reason about, and respectively easy to update and read.

Don’t worry about calling multiple useState() to create state variables for each concern.

Note, however, that if you have way too much useState() variables, there’s a good chance that your component violates the Single Responsibility Principle. Just split such components into smaller ones.

2. Extract complex state logic

Extract complex state logic into a custom hook. 

Would it make sense to keep complex state operations within the component?

The answer is in fundamentals (as usually happens).

React hooks are created to isolate the component from complex state management and side effects. So, since the component should be concerned only about the elements to render and some event listeners to attach, the complex state logic should be extracted into a custom hook.

Let’s consider a component that manages a list of products. The user can add new product names. The constraint is that product names have to be unique .

The first attempt is to keep the setter of product names list state directly inside the component:

function ProductsList() {
  const [names, setNames] = useState([]);  const [newName, setNewName] = useState('');

  const map = name => <div>{name}</div>;

  const handleChange = event => setNewName(event.target.value);
  const handleAdd = () => {    const s = new Set([...names, newName]);    setNames([...s]);  };
  return (
    <div className="products">
      {names.map(map)}
      <input type="text" onChange={handleChange} />
      <button onClick={handleAdd}>Add</button>
    </div>
  );
}

names state variable holds the product names. When the Add button is clicked, addNewProduct() event handler is invoked.

Inside addNewProduct() , a Set object is used to keep the product names unique. Should the component be concerned about this implementation detail? Nope.

It would be better to isolate the complex state setter logic into a custom hook. Let’s do that.

The new custom hook useUnique() takes care of keeping the items unique:

// useUnique.js
export function useUnique(initial) {
  const [items, setItems] = useState(initial);
  const add = newItem => {
    const uniqueItems = [...new Set([...items, newItem])];
    setItems(uniqueItems);
  };
  return [items, add];
};

Having the custom state management extracted into a hook, the ProductsList component becomes much lighter:

import { useUnique } from './useUnique';

function ProductsList() {
  const [names, add] = useUnique([]);  const [newName, setNewName] = useState('');

  const map = name => <div>{name}</div>;

  const handleChange = event => setNewName(e.target.value);
  const handleAdd = () => add(newName);
  return (
    <div className="products">
      {names.map(map)}
      <input type="text" onChange={handleChange} />
      <button onClick={handleAdd}>Add</button>
    </div>
  );
}

const [names, addName] = useUnique([]) is what enables the custom hook. The component is no longer cluttered with complex state management.

If you’d like to add a new name to the list, you only have to invoke add('New Product Name') .

Bottom line, the benefits of extracting the complex state management into a custom hook are:

  • The component becomes free of state management details
  • The custom hook can be reused
  • The custom hook can be easily tested in isolation

3. Extract multiple state operations

Extract multiple state operations into a reducer.

Continuing the example with ProductsList , let’s introduce a Delete operation, which deletes a product name from the list.

Now you have to code 2 operations: adding and deleting products. The handle these operations, it makes sense to create a reducer and make the component free of state management logic.

Again, this approach fits the idea of hooks: extract the complex state management out of the components.

Here’s a possible implementation of the reducer that adds and deletes products:

function uniqueReducer(state, action) {
  switch (action.type) {
    case 'add':
      return [...new Set([...state, action.name])];
    case 'delete':
      return state.filter(name => name === action.name);
    default:
      throw new Error();
  }
}

Then uniqueReducer() can be used inside the products list by invoking React’s useReducer() hook:

function ProductsList() {
  const [names, dispatch] = useReducer(uniqueReducer, []);  const [newName, setNewName] = useState('');

  const handleChange = event => setNewName(event.target.value);

  const handleAdd = () => dispatch({ type: 'add', name: newName });
  const map = name => {
    const delete = () => dispatch({ type: 'delete', name });    return (
      <div>
        {name}
        <button onClick={delete}>Delete</button>
      </div>
    );
  }

  return (
    <div className="products">
      {names.map(map)}
      <input type="text" onChange={handleChange} />
      <button onClick={handleAdd}>Add</button>
    </div>
  );
}

const [names, dispatch] = useReducer(uniqueReducer, []) enables uniqueReducer . names is the state variable holding the product names, and dispatch is a function to be called using an action object.

When Add button is clicked, the handler invokes dispatch({ type: 'add', name: newName }) . Dispatching an add action makes the reducer uniqueReducer add a new product name to the state.

In the same way, when Delete button is clicked, the handler invokes dispatch({ type: 'delete', name }) . Dispatching a remove action removes the product name from the state of names.

Interestingly, the reducer is a special case of Command design pattern .

4. Conclusion

A state variable should be responsible for one concern.

If the state has a complicated update logic, extract this logic out of the component into a custom hook.

Same way, if the state requires multiple operations, use a reducer to incorporate these operations.

No matter what rule you use, the state should be as simple and decoupled as possible. The component should not be cluttered with the details of how the state is updated: these should be a part of a custom hook or a reducer.

Confirming to these 3 simple rules will make your state logic easy to understand, maintain, and test.


以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持 码农网

查看所有标签

猜你喜欢:

本站部分资源来源于网络,本站转载出于传递更多信息之目的,版权归原作者或者来源机构所有,如转载稿涉及版权问题,请联系我们

矩阵论

矩阵论

方保镕 / 清华大学出版社 / 2004-1 / 39.00元

本书比较全面、系统地介绍了矩阵的基本理论、方法及其应用。全书分上、下两篇,共10章,分别介绍了线性空间与线性算子,内积空间与等积变换,λ矩陈与若尔当标准形,赋范线性空间与矩阵范数,矩阵的微积分运算及其应用,广义逆矩阵及其应用,矩阵的分解,矩阵的克罗内克积、阿达马积与反积,几类特殊矩阵(如:非负矩阵与正矩阵、循环矩阵与素矩阵、随机矩阵和双随机矩阵、单调矩阵、M矩阵与H矩阵、T矩阵与汉大象尔矩阵等),......一起来看看 《矩阵论》 这本书的介绍吧!

HTML 压缩/解压工具
HTML 压缩/解压工具

在线压缩/解压 HTML 代码

Base64 编码/解码
Base64 编码/解码

Base64 编码/解码

RGB HSV 转换
RGB HSV 转换

RGB HSV 互转工具