Sobes.tech
Back to questions
Middle+
4

import React, { useState, useEffect, useRef, useContext, useReducer, useMemo } from 'react'; const someFunc = (arr) => { return [...arr].sort((a, b) => a - b); }; const LazyInit = (props) => { const [arr, setArr] = useState(() => someFunc(props.arr)); const onClick = () => { setArr((prev) => prev.concat(prev.length + 1)) } return ( <> <button onClick={onClick}> Increment </button> <ul> {arr.map((n, index) => <li key={index}>{n}</li>)} </ul> </> ) } export default () => <LazyInit arr={[1, 2, 3]} />; --- The conversation on the right side discusses: - Using TypeScript with generics: "<T extends object, K extends keyof T>" - Checking if data is a string with a type guard: "isString(data: unknown): data is string { ... }" - Using useState with a function to avoid re-running on every render: ``` const [state, setState] = useState(getComplexData); // or const [state, setState] = useState(() => getComplexData()); ``` - The function passed to useState will run only once on component mount. - Warning about mutating state directly on line 11 with prev.concat. --- The technical task is to analyze and possibly fix or optimize the React component code, especially focusing on: - Proper lazy initialization of state with useState. - Avoiding direct mutation of state. - Understanding TypeScript generics and type guards. - Ensuring the component updates state correctly on button click.

Companies where asked
IT OneIT One

Answer from AI

sobes.tech AI

В представленном React-компоненте LazyInit используется ленивое инициализирование состояния через функцию в useState, что правильно, так как функция someFunc вызывается только один раз при монтировании компонента.

Однако есть важный момент с обновлением состояния в обработчике onClick:

const onClick = () => {
  setArr((prev) => prev.concat(prev.length + 1));
}

Здесь используется метод concat, который не мутирует исходный массив, а возвращает новый — это корректно и не нарушает иммутабельность состояния. Поэтому предупреждение о мутации состояния не обосновано в данном случае.

Если бы использовался метод push, то это была бы мутация, и нужно было бы избегать этого.

Возможные улучшения:

  • Для ключей в списке лучше использовать уникальные значения, а не индекс, чтобы избежать проблем с рендерингом при изменении массива:
{arr.map((n) => <li key={n}>{n}</li>)}
  • Если props.arr может изменяться, а состояние нужно инициализировать заново, стоит добавить эффект useEffect для обновления состояния при изменении пропса.

  • В части TypeScript:

    • Использование дженериков <T extends object, K extends keyof T> позволяет создавать обобщённые функции или компоненты, которые работают с объектами и их ключами, обеспечивая типовую безопасность.
    • Типовой гард isString(data: unknown): data is string помогает уточнять типы в рантайме, что полезно для безопасной работы с данными неизвестного типа.

В целом, компонент корректен, и ленивое инициализирование состояния реализовано правильно.