2

map我有一个通过函数在页面加载时呈现的图像列表。我希望能够选择一张或多张图像,并突出显示它们。此外,所选图像的标题将显示在页面顶部。我遇到的问题是每次选择图像时列表似乎都会重新呈现。这是组件:

export default function App() {
  const [images, setImages] = React.useState<Image[]>([]);
  const [selected, setSelected] = React.useState<boolean>(false);
  const [imageTitles, setImageTitles] = React.useState<string[]>([]);

  const handleImageSelection = (title: string, index: number) => {
    setSelected(!selected);
    const imageExists = imageTitles.indexOf(title) !== -1
    if (!imageExists) {
      setImageTitles([...imageTitles, title]);
    } else {
      setImageTitles(imageTitles.filter(str => str !== title));
    }
  }

  return (
    <div>
      {console.log(images)} // The images array renders every time
      <div>{imageTitles.map(title => <p>{title}</p>)}</div>

      <section className="images">{
        images.map(({ title, image}, index) =>
          <img
            style={{
              marginBottom: 4,
              border: imageTitles.indexOf(title) !== -1 ?
                "4px solid blue" : "", // A selected image is highlighted
              borderRadius: 16
            }}
            src={image}
            alt={image}
            onClick={() => handleImageSelection(title, index)}
          />
        )
      }</section>
    </div>
  );
}

我想知道是否是因为每次选择/取消选择图像时我都在更改imageTitles数组的大小(以及因此的值)。index

我也试过useCallback这样:

const handleImageSelection = React.useCallback((title: string, index: number) => {
  setSelected(!selected);
  const imageExists = imageTitles.indexOf(title) !== -1
  if (!imageExists) {
    setImageTitles([...imageTitles, title]);
  } else {
    setImageTitles(imageTitles.filter(str => str !== title));
  }
}, [selected, imageTitles])

但它似乎没有奏效。我的猜测是因为imagetitles每次都在变化。

那么,是否有可能(出于性能原因)避免每次选择/取消选择图像时重新渲染图像列表?

4

1 回答 1

1

由于您在 useCallback 的依赖项中添加了 selected 和 imageTitles,因此每次调用 useCallback 时都会重新创建它,因为它本身会设置 selected 和 imageTitles 状态

这里的解决方案是使用 setState 回调模式并将空数组作为依赖项传递给 useCallback

const handleImageSelection = React.useCallback((title: string, index: number) => {
  setSelected(prevSelected => !prevSelected);


    setImageTitles(prevImageTitles => {
        const imageExists = prevImageTitles.indexOf(title) !== -1;
        if (!imageExists) { 
           return [...prevImageTitles, title];
        } else {
           return prevImageTitles.filter(str => str !== title));
        }
    });

}, []);

另请注意,每次您设置正常行为的状态时,该列表都会重新呈现。但是 react 会优化渲染,并且只会重新渲染那些需要更改的元素。

这里要注意的另一件事是,您必须将 key prop 传递给映射值,以便 React 进一步优化重新渲染

return (
    <div>
      <div>{imageTitles.map(title => <p key={title}>{title}</p>)}</div>

      <section className="images">{
        images.map(({ title, image}, index) =>
          <img
            key={title} // any unique value here. If you don't have anything use index
            style={{
              marginBottom: 4,
              border: imageTitles.indexOf(title) !== -1 ?
                "4px solid blue" : "", // A selected image is highlighted
              borderRadius: 16
            }}
            src={image}
            alt={image}
            onClick={() => handleImageSelection(title, index)}
          />
        )
      }</section>
    </div>
  );
于 2020-05-25T13:09:14.783 回答