RN开发搬砖经验之—在React 函数式组件别一把梭useState得考虑下useRef

本文主要是介绍RN开发搬砖经验之—在React 函数式组件别一把梭useState得考虑下useRef,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

在这里插入图片描述

最近在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 元素时(如

),你可以通过 myRef.current 访问到这个 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;

这篇关于RN开发搬砖经验之—在React 函数式组件别一把梭useState得考虑下useRef的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/778825

相关文章

Java实现Excel与HTML互转

《Java实现Excel与HTML互转》Excel是一种电子表格格式,而HTM则是一种用于创建网页的标记语言,虽然两者在用途上存在差异,但有时我们需要将数据从一种格式转换为另一种格式,下面我们就来看看... Excel是一种电子表格格式,广泛用于数据处理和分析,而HTM则是一种用于创建网页的标记语言。虽然两

基于Python开发电脑定时关机工具

《基于Python开发电脑定时关机工具》这篇文章主要为大家详细介绍了如何基于Python开发一个电脑定时关机工具,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录1. 简介2. 运行效果3. 相关源码1. 简介这个程序就像一个“忠实的管家”,帮你按时关掉电脑,而且全程不需要你多做

Java中的Opencv简介与开发环境部署方法

《Java中的Opencv简介与开发环境部署方法》OpenCV是一个开源的计算机视觉和图像处理库,提供了丰富的图像处理算法和工具,它支持多种图像处理和计算机视觉算法,可以用于物体识别与跟踪、图像分割与... 目录1.Opencv简介Opencv的应用2.Java使用OpenCV进行图像操作opencv安装j

vue解决子组件样式覆盖问题scoped deep

《vue解决子组件样式覆盖问题scopeddeep》文章主要介绍了在Vue项目中处理全局样式和局部样式的方法,包括使用scoped属性和深度选择器(/deep/)来覆盖子组件的样式,作者建议所有组件... 目录前言scoped分析deep分析使用总结所有组件必须加scoped父组件覆盖子组件使用deep前言

VUE动态绑定class类的三种常用方式及适用场景详解

《VUE动态绑定class类的三种常用方式及适用场景详解》文章介绍了在实际开发中动态绑定class的三种常见情况及其解决方案,包括根据不同的返回值渲染不同的class样式、给模块添加基础样式以及根据设... 目录前言1.动态选择class样式(对象添加:情景一)2.动态添加一个class样式(字符串添加:情

基于Qt Qml实现时间轴组件

《基于QtQml实现时间轴组件》时间轴组件是现代用户界面中常见的元素,用于按时间顺序展示事件,本文主要为大家详细介绍了如何使用Qml实现一个简单的时间轴组件,需要的可以参考下... 目录写在前面效果图组件概述实现细节1. 组件结构2. 属性定义3. 数据模型4. 事件项的添加和排序5. 事件项的渲染如何使用

Oracle的to_date()函数详解

《Oracle的to_date()函数详解》Oracle的to_date()函数用于日期格式转换,需要注意Oracle中不区分大小写的MM和mm格式代码,应使用mi代替分钟,此外,Oracle还支持毫... 目录oracle的to_date()函数一.在使用Oracle的to_date函数来做日期转换二.日

基于Qt开发一个简单的OFD阅读器

《基于Qt开发一个简单的OFD阅读器》这篇文章主要为大家详细介绍了如何使用Qt框架开发一个功能强大且性能优异的OFD阅读器,文中的示例代码讲解详细,有需要的小伙伴可以参考一下... 目录摘要引言一、OFD文件格式解析二、文档结构解析三、页面渲染四、用户交互五、性能优化六、示例代码七、未来发展方向八、结论摘要

React实现原生APP切换效果

《React实现原生APP切换效果》最近需要使用Hybrid的方式开发一个APP,交互和原生APP相似并且需要IM通信,本文给大家介绍了使用React实现原生APP切换效果,文中通过代码示例讲解的非常... 目录背景需求概览技术栈实现步骤根据 react-router-dom 文档配置好路由添加过渡动画使用

在 VSCode 中配置 C++ 开发环境的详细教程

《在VSCode中配置C++开发环境的详细教程》本文详细介绍了如何在VisualStudioCode(VSCode)中配置C++开发环境,包括安装必要的工具、配置编译器、设置调试环境等步骤,通... 目录如何在 VSCode 中配置 C++ 开发环境:详细教程1. 什么是 VSCode?2. 安装 VSCo