最近在fix一些bug中,发现在函数式组件中不区别场景,任何函数式组件中的变量都是使用useState,然后没有考虑到useState是异步更新值的,导致各种离奇的BUG出现!另外看到相关代码中出现大量的setTimeout操作,估计想用它来规避useState是异步更新值的行为,这种情况下代码就更容易出bug,也很难维护了!
当使用 useState 时,我们如果不正确地处理异步操作,可能会导致意料之外的行为。useState 的调用操作是同步的,其它是异步更新的,如果你在事件处理函数或其他异步回调中直接更新状态,并期望状态立即改变,这可能会导致问题。下面是一个这样的示例代码:
import React, { useState } from 'react';
function BuggyComponent() {
const [count, setCount] = useState(0);
const handleClick = async () => {
// 假设这是一个异步操作,比如网络请求
const response = await fetchSomeData();
// 基于异步操作的结果来更新状态
setCount(prevCount => prevCount + 1);
// 假设我们立即需要使用更新后的状态
console.log(count); // 这里可能会输出旧的值,而不是更新后的值
};
return (
<div>
<p>Count: {count}</p>
<button onClick={handleClick}>Increment Count</button>
</div>
);
}
// 假设的异步函数,用于模拟网络请求
async function fetchSomeData() {
return new Promise(resolve => {
setTimeout(() => {
resolve('Data fetched!');
}, 1000);
});
}
export default BuggyComponent;
useState的特性
useState是React函数组件中用于管理状态(state)的Hook。它接受一个初始状态,并返回一个数组,其中包含当前状态和一个函数,用于更新当前状态。以下是useState的主要特性和注意点:
特性:
响应式:useState是响应式的,当状态改变时,它会触发组件的重新渲染。
接受任意JavaScript值:useState可以接受任何JavaScript值作为初始状态。
返回数组:useState返回一个数组,其中第一个元素是当前的状态值,第二个元素是一个函数,用于更新该状态值。
使用注意事项:
useState的位置:useState应该被放在函数组件的顶层,即在任何条件语句或循环之前。这是因为useState在每次渲染时都会执行,如果在条件语句或循环中使用useState,可能会导致不可预知的结果。
状态的更新:当使用useState返回的更新函数来改变状态时,如果传入的新值与旧值相同(使用Object.is进行浅比较),那么不会触发组件的重新渲染。此外,useState的更新函数不会与之前的状态进行合并,而是直接替换掉之前的状态。因此,在更新对象或数组时,需要注意保存之前的状态。
避免直接修改状态:不应该直接修改useState返回的状态值,而应该使用更新函数来更新状态。这是因为直接修改状态值不会触发组件的重新渲染,这可能会导致视图与状态不一致。
初始状态的设置:useState的初始状态只在组件的第一次渲染时设置。如果初始状态依赖于组件的props,那么应该使用useEffect来更新状态。
useRef的特性
useRef 是 React 的一个 Hook,它返回一个可变的 ref 对象,其 .current 属性可以被设置为一个 DOM 元素或者任何你想要保持引用的值。useRef 有一些独特的特性和使用注意事项:
特性:
稳定性:useRef 创建的 ref 对象在组件的整个生命周期内保持不变。
不触发重新渲染:修改 ref.current 的值不会引发组件的重新渲染。
通用性:ref.current 可以保存任何类型的值,不仅仅是 DOM 元素。
访问性:即使在函数组件的每次渲染中,你都可以通过 ref.current 访问到最新的值。
使用注意事项:
1、不作为依赖项:ref.current 不应作为 useEffect、useMemo 或 useCallback 等其他 Hooks 的依赖项,因为 React 不会跟踪 ref.current 的变化来触发重新渲染。
2、不用于状态管理:由于修改 ref.current 不会触发重新渲染,因此不应使用 useRef 来管理需要在状态变化时更新视图的状态。这是 useState 的主要用途。
3、初始值设置:你可以在创建 useRef 时为其提供一个初始值,但这个初始值只在第一次渲染时设置。之后,你可以通过直接赋值来更改 ref.current 的值。
4、访问 DOM 元素:当 ref 被绑定到一个 DOM 元素时(如
5、保存可变对象:由于 useRef 创建的 ref 对象在组件生命周期内保持不变,并且修改其 .current 属性不会触发重新渲染,因此它非常适合用于保存可变对象,如定时器 ID、订阅 ID 或可变的数据结构
两者的使用场景
useState和useRef都是React的Hooks,但它们有不同的使用场景和目的。
1、useState主要用于在函数组件中管理状态。当状态发生变化时,组件会重新渲染。它是异步的,同一个函数内设置的,不能实时获取到最新的值。useState的使用场景通常包括需要在状态改变时重新渲染视图的场景。例如,你可以使用useState来创建一个计数器,当计数器的值变化时,整个组件会重新渲染,显示新的计数器值。
2、useRef则主要用于访问DOM元素或者保存对可变对象的引用,这种引用不会触发组件重新渲染。useRef返回的可变对象在组件的整个生命周期内保持不变,且设置的值是同步的,同一个函数内设置的,能实时获取到最新的值。useRef的一个常见用例是将ref对象绑定到DOM元素上,以便在必要时访问DOM元素的属性和方法。此外,useRef也可以用于保存可变对象的引用,而不影响视图的更新。
总的来说,useState主要用于数据的变化和视图的更新,而useRef则主要用于访问和交互不会触发渲染的对象。
需要注意的是,ref.current不可以作为其他hooks(如useMemo, useCallback, useEffect)的依赖项,因为ref.current的值发生变更并不会造成re-render,Reactjs并不会跟踪ref.current的变化。同时,变量(组件内)在每次组件重新渲染的时候都会被重新进行赋值为初始值,所以如果你想要保留之前操作的状态的话就不要使用变量。
比如加列表分页加载时分页变量,其实是可以用useRef来替换useState的,特别是当你的分页变量在更改时,就需要访问(同步获取更新后的值)且分页变量不希望触发组件重新渲染(多数情况下分页变量是不需要触发重新渲染的)这种场景时,就特别需要useRef了。
使用useState的示例如下:
import React, { useState, useEffect } from 'react';
import { FlatList, View, Text, ActivityIndicator } from 'react-native';
const PageSize = 10; // 每页的项目数量
const MyFlatList = () => {
const [loading, setLoading] = useState(false);
const [dataList, setDataList] = useState([]);
const [hasMore, setHasMore] = useState(true);
const [page, setPage] = useState(1);
// 加载更多数据的函数
const loadMoreData = async () => {
if (loading || !hasMore) return;
setLoading(true);
// 模拟网络请求获取下一页数据
const newData = await fetchData(page);
// 更新数据列表和页面
setDataList(prevData => [...prevData, ...newData]);
setPage(prevPage => prevPage + 1);
// 根据返回的数据判断是否还有更多页面
setHasMore(newData.length === PageSize);
setLoading(false);
};
// 模拟从服务器获取数据的函数
const fetchData = async (page) => {
// 假设这是从服务器获取的数据
// 在实际应用中,你会发送一个网络请求到服务器,并处理响应
const start = (page - 1) * PageSize + 1;
const end = start + PageSize;
return Array.from({ length: PageSize }, (_, i) => `Item ${start + i}`);
};
return (
<FlatList
data={dataList}
keyExtractor={item => item.toString()}
renderItem={({ item }) => <Text>{item}</Text>}
onEndReached={loadMoreData}
onEndReachedThreshold={0.5}
ListFooterComponent={
loading ? (
<View style={{ paddingVertical: 20 }}>
<ActivityIndicator size="large" />
</View>
) : null
}
/>
);
};
export default MyFlatList;
改成分页变量用useRef的示例如下:
import React, { useState, useEffect, useRef } from 'react';
import { FlatList, View, Text, ActivityIndicator } from 'react-native';
const PageSize = 10; // 每页的项目数量
const MyFlatList = () => {
const [loading, setLoading] = useState(false);
const [dataList, setDataList] = useState([]);
const [hasMore, setHasMore] = useState(true);
const pageRef = useRef(1); // 使用 ref 来存储当前的页码
// 加载更多数据的函数
const loadMoreData = async () => {
if (loading || !hasMore) return;
setLoading(true);
// 模拟网络请求获取下一页数据
const currentPage = pageRef.current;
const newData = await fetchData(currentPage);
// 更新数据列表和页码
setDataList(prevData => [...prevData, ...newData]);
pageRef.current = currentPage + 1; // 更新 ref 中的页码
// 根据返回的数据判断是否还有更多页面
setHasMore(newData.length === PageSize);
setLoading(false);
};
// 模拟从服务器获取数据的函数
const fetchData = async (page) => {
// 假设这是从服务器获取的数据
// 在实际应用中,你会发送一个网络请求到服务器,并处理响应
const start = (page - 1) * PageSize + 1;
const end = start + PageSize;
return Array.from({ length: PageSize }, (_, i) => `Item ${start + i}`);
};
return (
<FlatList
data={dataList}
keyExtractor={item => item.toString()}
renderItem={({ item }) => <Text>{item}</Text>}
onEndReached={loadMoreData}
onEndReachedThreshold={0.5}
ListFooterComponent={
loading ? (
<View style={{ paddingVertical: 20 }}>
<ActivityIndicator size="large" />
</View>
) : null
}
/>
);
};
export default MyFlatList;
使用useRef
类组件
代码示例如下:
import React from 'react';
class MyComponent extends React.Component {
constructor(props) {
super(props);
// 初始化实例属性来模拟 ref
this.myRef = React.createRef();
}
componentDidMount() {
// 类似于 useEffect,在组件挂载后访问 DOM 元素
const node = this.myRef.current;
if (node) {
console.log(node); // 输出 DOM 元素
}
}
render() {
return (
<div ref={this.myRef}>
Hello, World!
</div>
);
}
}
export default MyComponent;
函数式组件
代码示例如下:
import React, { useRef, useEffect } from 'react';
function MyFunctionalComponent() {
// 创建一个 ref 来保存对 DOM 元素的引用
const myRef = useRef(null);
// 使用 useEffect 在组件挂载后和卸载前执行操作
useEffect(() => {
// 组件挂载后
const node = myRef.current;
if (node) {
console.log(node); // 输出 DOM 元素
// 可以执行其他操作,比如设置焦点、监听事件等
}
// 返回一个清理函数,在组件卸载前执行
return () => {
// 组件卸载前,可以执行清理操作,比如移除事件监听器
};
}, []); // 注意依赖项数组为空,确保只在挂载和卸载时运行
return (
<div ref={myRef}>
Click me
</div>
);
}
export default MyFunctionalComponent;