下面的代码实现了一个简易的登录功能(为了减少代码量,去掉了密码)。
import React, { useState } from 'react';
const api = {
login(username) {
console.log('username', username);
},
};
function Login() {
const [username, setUsername] = useState('');
const onSubmit = () => {
if (!username) {
alert('Please input username');
return;
}
api.login(username,);
};
return (
Login
setUsername(e.target.value)}
/>
);
}
export default Login;
现在我们要增加一个需求:
监听页面上的回车事件,然后发起登录,代码如下:
useEffect(() => {
const onKeyup = (e) => {
if (e.key === 'Enter') {
onSubmit();
}
}
window.addEventListener('keyup', onKeyup);
}, []);
代码看起来没问题,但运行一下,我们就会发现并没有按照预期发展,在输入框已经输入用户名的前提下,页面上还是会一直提示让我们输入用户名。
什么原因?
问题出在onKeyup
内部的onSubmit
,由于onSubmit
所在的useEffect
没有依赖,所以只会在初始化执行一次,onSubmit
内部的username
也就会一直处于初次渲染的状态,值为空字符。
如何解决?
将onSubmit
加入useEffect
的依赖中。
useEffect(() => {
const onKeyup = (e) => {
if (e.key === 'Enter') {
onSubmit();
}
}
window.addEventListener('keyup', onKeyup);
}, [onSubmit]);
依赖加好了,但是看着上面的代码,我们又会发现一个问题,由于react
会在每次渲染的时候重新创建组件内非hooks
的值,这就导致了每次渲染的onSubmit
的值发生改变,进而导致监听onSubmit
的useEffect重复执行。
如何解决?
给onSubmit
加上useCallback
,将username
加入到useCallback
的依赖中。
const onSubmit = useCallback(() => {
if (!username) {
alert('Please input username');
return;
}
login(username);
}, [username]);
这样,在每次组件渲染时,只有当username
的值发生改变,onSubmit
的值才会跟着发生改变,从而解决了useEffect
反复执行的问题。
我们再运行一下,又有新问题了!
当我们输入的username
长度超过1
时,会发现login
接口被调用了多次。
什么原因?
问题来到了useCallbak
,每次username
发生变化的时候,onSubmit
都会被更新,从而导致useEffect
为每次username
的改变都执行了一次,也就多次绑定了onKeyup
了,结果是多次调用onSubmit
。
怎么解决?
利用useEffect
的cleanup
。
在cleanup
里面解绑onKeyup
。这样就不会存在多次绑定的onKeyup
被执行的情况了,只会存在一个最新的onKeyup
绑定。
useEffect(() => {
const onKeyup = (e) => {
if (e.key === 'Enter') {
onSubmit();
}
}
window.addEventListener('keyup', onKeyup);
return () => {
window.removeEventListener('keyup', onKeyup);
}
}, [onSubmit]);
至此,代码工作的完好。
完整的代码如下:
import React, { useState, useEffect, useCallback } from 'react';
const login = (username) => {
console.log('username', username);
};
function Login() {
const [username, setUsername] = useState('');
const onSubmit = useCallback(() => {
if (!username) {
alert('Please input username');
return;
}
login(username);
}, [username]);
useEffect(() => {
const onKeyup = (e) => {
if (e.key === 'Enter') {
onSubmit();
}
}
window.addEventListener('keyup', onKeyup);
return () => {
window.removeEventListener('keyup', onKeyup);
}
}, [onSubmit]);
return (
Login
setUsername(e.target.value)}
/>
);
}
export default Login;