反应钩子 useState() 的行为真的很奇怪
Really weird behavior of the react hook useState()
我目前正在使用 Swiper 编写一个用于简单幻灯片放映的 React 应用程序,一切正常。我现在正尝试通过使用 setActiveSlide 函数显示当前幻灯片来改进导航。第一次加载后,它工作正常。单击调用 setActiveSlide 的按钮后,story 对象设置为 null,这会导致错误(Uncaught TypeError: Cannot read 属性 'slideNext' of null)当您单击 Previous 或 Next 按钮一秒钟时间。为什么调用setActiveSlide后story对象设置为null?请帮助
这是应用程序的简化版本
import React, { useState, useEffect, useRef } from "react";
import Swiper from "swiper";
const Slideshow = props => {
var swiper = useRef();
var story = null;
const [activeSlide, setActiveSlide] = useState("");
useEffect(() => {
story = new Swiper(swiper.current, {
loop: false,
speed: 1100
});
}, []);
const prevSlide = () => {
setActiveSlide("PREV");
story.slidePrev();
};
const nextSlide = () => {
setActiveSlide("NEXT");
story.slideNext();
};
return (
<div className="container-story" ref={swiper}>
<div className="container-story-wrapper swiper-wrapper">
{props.children}
</div>
<div className="navigation">
<button onClick={prevSlide}> Previous </button>
<button onClick={nextSlide}> Next </button>
{activeSlide}
</div>
<div className="lightbox-container"></div>
</div>
);
};
export default Slideshow;
这是一个函数组件,所以如果您希望一个变量在多次渲染中保持不变,那么您不能像在 var story = null
中那样在本地声明它。您的第一个渲染将设置它,但是当组件重新渲染时,您将其设置回 null。
处理这个问题的正确方法是使用 Reacts useRef
钩子。这是一种让变量在多次渲染中保持不变的方法。
要声明它:
const _story = useRef(null);
到 read/write 到 _story
变量,您将需要访问 _story.current
。所以在你的 useEffect
安装中你应该使用这个:
_story.current = new Swiper(swiper.current, {
loop: false,
speed: 1100
});
在你的处理程序中:
_story.current.slideNext();
您可以阅读更多关于 useRef
here。
所以问题是
从 React Hook useEffect 内部对 'story' 变量的赋值将在每次渲染后丢失。为了随时间保留值,将其存储在 useRef Hook 中并将可变值保留在“.current”属性 中。否则,您可以直接在 useEffect.
内部移动此变量
所以你需要更新如下
import React, { useState, useEffect, useRef } from "react";
import "./styles.css";
import Swiper from "swiper";
const Slideshow = props => {
var swiper = useRef();
var story = useRef(null);
const [activeSlide, setActiveSlide] = useState("");
useEffect(() => {
story.current = new Swiper(swiper.current, {
loop: false,
speed: 1100
});
}, []);
const prevSlide = () => {
setActiveSlide("PREV");
story.current.slidePrev();
};
const nextSlide = () => {
setActiveSlide("NEXT");
story.current.slideNext();
};
return (
<div className="container-story" ref={swiper}>
<div className="container-story-wrapper swiper-wrapper">
{props.children}
</div>
<div className="navigation">
<button onClick={prevSlide}> Previous </button>
<button onClick={nextSlide}> Next </button>
{activeSlide}
</div>
<div className="lightbox-container" />
</div>
);
};
export default Slideshow;
我目前正在使用 Swiper 编写一个用于简单幻灯片放映的 React 应用程序,一切正常。我现在正尝试通过使用 setActiveSlide 函数显示当前幻灯片来改进导航。第一次加载后,它工作正常。单击调用 setActiveSlide 的按钮后,story 对象设置为 null,这会导致错误(Uncaught TypeError: Cannot read 属性 'slideNext' of null)当您单击 Previous 或 Next 按钮一秒钟时间。为什么调用setActiveSlide后story对象设置为null?请帮助
这是应用程序的简化版本
import React, { useState, useEffect, useRef } from "react";
import Swiper from "swiper";
const Slideshow = props => {
var swiper = useRef();
var story = null;
const [activeSlide, setActiveSlide] = useState("");
useEffect(() => {
story = new Swiper(swiper.current, {
loop: false,
speed: 1100
});
}, []);
const prevSlide = () => {
setActiveSlide("PREV");
story.slidePrev();
};
const nextSlide = () => {
setActiveSlide("NEXT");
story.slideNext();
};
return (
<div className="container-story" ref={swiper}>
<div className="container-story-wrapper swiper-wrapper">
{props.children}
</div>
<div className="navigation">
<button onClick={prevSlide}> Previous </button>
<button onClick={nextSlide}> Next </button>
{activeSlide}
</div>
<div className="lightbox-container"></div>
</div>
);
};
export default Slideshow;
这是一个函数组件,所以如果您希望一个变量在多次渲染中保持不变,那么您不能像在 var story = null
中那样在本地声明它。您的第一个渲染将设置它,但是当组件重新渲染时,您将其设置回 null。
处理这个问题的正确方法是使用 Reacts useRef
钩子。这是一种让变量在多次渲染中保持不变的方法。
要声明它:
const _story = useRef(null);
到 read/write 到 _story
变量,您将需要访问 _story.current
。所以在你的 useEffect
安装中你应该使用这个:
_story.current = new Swiper(swiper.current, {
loop: false,
speed: 1100
});
在你的处理程序中:
_story.current.slideNext();
您可以阅读更多关于 useRef
here。
所以问题是
从 React Hook useEffect 内部对 'story' 变量的赋值将在每次渲染后丢失。为了随时间保留值,将其存储在 useRef Hook 中并将可变值保留在“.current”属性 中。否则,您可以直接在 useEffect.
内部移动此变量所以你需要更新如下
import React, { useState, useEffect, useRef } from "react";
import "./styles.css";
import Swiper from "swiper";
const Slideshow = props => {
var swiper = useRef();
var story = useRef(null);
const [activeSlide, setActiveSlide] = useState("");
useEffect(() => {
story.current = new Swiper(swiper.current, {
loop: false,
speed: 1100
});
}, []);
const prevSlide = () => {
setActiveSlide("PREV");
story.current.slidePrev();
};
const nextSlide = () => {
setActiveSlide("NEXT");
story.current.slideNext();
};
return (
<div className="container-story" ref={swiper}>
<div className="container-story-wrapper swiper-wrapper">
{props.children}
</div>
<div className="navigation">
<button onClick={prevSlide}> Previous </button>
<button onClick={nextSlide}> Next </button>
{activeSlide}
</div>
<div className="lightbox-container" />
</div>
);
};
export default Slideshow;