在 React Hooks 中是否有设置状态的通用方法?如何管理多个状态?

IT技术 javascript reactjs react-hooks
2021-02-15 20:01:56

我有一个问题,我是否可以useState在 React Hooks 中使用泛型,就像我可以在 React Components 中同时管理多个状态一样?

state = {
  input1: "",
  input2: "",
  input3: ""
  // .. more states
};

handleChange = (event) => {
  const { name, value } = event.target;
  this.setState({
    [name]: value,
  });
};
3个回答

是的,使用钩子,您可以通过三种方式管理复杂的状态(无需第 3 方库),其中的主要原因是管理状态 ID及其相应的元素。

  1. 管理具有多个状态的单个对象(请注意,数组是一个对象)。
  2. 使用useReducerif ( 1) 太复杂。
  3. useState对每个键值对使用多个(考虑它的可读性和维护性)。

看看这个:

// Ids-values pairs.
const complexStateInitial = {
  input1: "",
  input2: "",
  input3: ""
  // .. more states
};

function reducer(state, action) {
  return { ...state, [action.type]: action.value };
}

export default function App() {
  const [fromUseState, setState] = useState(complexStateInitial);

  // handle generic state from useState
  const onChangeUseState = (e) => {
    const { name, value } = e.target;
    setState((prevState) => ({ ...prevState, [name]: value }));
  };

  const [fromReducer, dispatch] = useReducer(reducer, complexStateInitial);

  // handle generic state from useReducer
  const onChangeUseReducer = (e) => {
    const { name, value } = e.target;
    dispatch({ type: name, value });
  };
 
  return (
    <>
      <h3>useState</h3>
      <div>
        {Object.entries(fromUseState).map(([key, value]) => (
          <input
            key={key}
            name={key}
            value={value}
            onChange={onChangeUseState}
          />
        ))}
        <pre>{JSON.stringify(fromUseState, null, 2)}</pre>
      </div>

      <h3>useReducer</h3>
      <div>
        {Object.entries(fromReducer).map(([key, value]) => (
          <input
            name={key}
            key={key}
            value={value}
            onChange={onChangeUseReducer}
          />
        ))}
        <pre>{JSON.stringify(fromReducer, null, 2)}</pre>
      </div>
    </>
  );
}

编辑处理多个状态

笔记

  • setState在类组件中找到方法不同useState不会自动合并更新对象。您可以通过将函数更新程序形式与对象传播语法相结合来复制此行为:
setState(prevState => {
  // Object.assign would also work
  return {...prevState, ...updatedValues};
});

请参阅React 文档

@DennisVash:我道歉。你是对的。我没有看到您的 setState 声明,并认为它与 OP 似乎在询问的类 setState 方法相同。我的困惑是您使用了相同的方法名称。
2021-04-17 20:01:56
@James:不要妄自菲薄。我认为我们实际上同意。您的自定义钩子答案最终可能是 OP 正在寻找的答案。
2021-04-17 20:01:56
setState来自useStateconst [state, setState] = useState(initialState);
2021-04-21 20:01:56
这不使用react-hooks。
2021-04-25 20:01:56
这是正确的,但不是使用 react hooks 的方法。
2021-05-10 20:01:56

做你想做的事情的正确方法是创建你自己的useState内部使用的钩子

下面是一个例子:

// This is your generic reusable hook.
const useHandleChange = (initial) => {
  const [value, setValue] = React.useState(initial);
  const handleChange = React.useCallback(
    (event) => setValue(event.target.value), // This is the meaty part.
    []
  );
  return [value, handleChange];
}

const App = () => {
  // Here we use the hook 3 times to show it's reusable.
  const [value1, handle1] = useHandleChange('one');
  const [value2, handle2] = useHandleChange('two');
  const [value3, handle3] = useHandleChange('three');
  return <div>
    <div>
      <input onChange={handle1} value={value1} />
      <input onChange={handle2} value={value2} />
      <input onChange={handle3} value={value3} />
    </div>
    <h2>States:</h2>
    <ul>
      <li>{value1}</li>
      <li>{value2}</li>
      <li>{value3}</li>
    </ul>
  </div>
}

ReactDOM.render(<App />, document.querySelector("#app"))
<script src="https://cdnjs.cloudflare.com/ajax/libs/react/16.8.6/umd/react.production.min.js"></script>
<script src="https://cdnjs.cloudflare.com/ajax/libs/react-dom/16.8.6/umd/react-dom.production.min.js"></script>
<div id="app"></div>

注意使用 React.useCallback 来阻止你的钩子在每次渲染时返回一个新的处理函数。(我们不需要指定setValue为依赖,因为 React 保证它永远不会改变)

我没有实际测试过这个,但它应该可以工作。

有关更多信息,请参阅https://reactjs.org/docs/hooks-reference.html#usestate

import React, {useState} from 'react';

const MyComponent = () => {
    const [name, setName] = useState('Default value for name');
    return (<div><button onClick={()=>setName('John Doe')}}>Set Name</button></div>);
};
export default MyComponent;
我不明白这如何回答问题?您刚刚给出了一个如何使用的示例,setStateOP 如何从中推断出如何动态设置它?
2021-05-06 20:01:56