WHAT - 通过 react-use 源码学习 React(Side-effects 篇)

2024-08-29 17:52

本文主要是介绍WHAT - 通过 react-use 源码学习 React(Side-effects 篇),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

目录

  • 一、官方介绍
    • 1. Sensors
    • 2. UI
    • 3. Animations
    • 4. Side-Effects
    • 5. Lifecycles
    • 6. State
    • 7. Miscellaneous
  • 二、源码学习
    • 示例:n. xx - yy
    • Side-effects - useAsync, useAsyncFn, and useAsyncRetry
      • useAsync
      • useAsyncFn
      • useAsyncRetry

一、官方介绍

Github 地址

react-use 是一个流行的 React 自定义 Hook 库,提供了一组常用的 Hook,以帮助开发者在 React 应用程序中更方便地处理常见的任务和功能。

官方将 react-use 的 Hook 分成了以下几个主要类别,以便更好地组织和查找常用的功能。每个类别涵盖了不同类型的 Hook,满足各种开发需求。以下是这些类别的详细说明:

1. Sensors

  • 功能: 主要涉及与浏览器或用户交互相关的传感器功能。
  • 示例:
    • useMouse: 获取鼠标位置。
    • useWindowSize: 获取窗口尺寸。
    • useBattery: 监控电池状态。

2. UI

  • 功能: 涉及用户界面相关的功能,如处理样式、显示和隐藏元素等。
  • 示例:
    • useClickAway: 监听点击事件以检测用户点击是否发生在组件外部。
    • useMeasure: 测量元素的大小和位置。
    • useDarkMode: 管理和检测暗模式状态。

3. Animations

  • 功能: 处理动画和过渡效果。
  • 示例:
    • useSpring: 使用 react-spring 处理动画效果。
    • useTransition: 使用 react-spring 处理过渡动画。

4. Side-Effects

  • 功能: 处理副作用相关的 Hook,包括数据获取、异步操作等。
  • 示例:
    • useAsync: 处理异步操作,如数据获取,并提供状态和结果。
    • useFetch: 简化数据获取操作。
    • useAxios: 使用 Axios 进行数据请求的 Hook。

5. Lifecycles

  • 功能: 处理组件生命周期相关的 Hook。
  • 示例:
    • useMount: 在组件挂载时执行的 Hook。
    • useUnmount: 在组件卸载时执行的 Hook。
    • useUpdate: 在组件更新时执行的 Hook。

6. State

  • 功能: 管理组件状态和相关逻辑。
  • 示例:
    • useState: 提供基本状态管理功能。
    • useReducer: 替代 useState 实现更复杂的状态逻辑。
    • useForm: 管理表单状态和验证。
    • useInput: 管理输入字段的状态。

7. Miscellaneous

  • 功能: 各种其他实用功能的 Hook,涵盖一些不容易归类到其他类别的功能。

这种分类方法使得 react-use 的 Hook 更加有组织和易于查找,帮助开发者快速找到需要的功能并有效地集成到他们的应用程序中。

二、源码学习

示例:n. xx - yy

something

使用

源码

解释

Side-effects - useAsync, useAsyncFn, and useAsyncRetry

resolves an async function.

useAsync

使用

import {useAsync} from 'react-use';const Demo = ({url}) => {const state = useAsync(async () => {const response = await fetch(url);const result = await response.text();return result}, [url]);return (<div>{state.loading? <div>Loading...</div>: state.error? <div>Error: {state.error.message}</div>: <div>Value: {state.value}</div>}</div>);
};

源码

import { DependencyList, useEffect } from 'react';
import useAsyncFn from './useAsyncFn';
import { FunctionReturningPromise } from './misc/types';export { AsyncState, AsyncFnReturn } from './useAsyncFn';export default function useAsync<T extends FunctionReturningPromise>(fn: T,deps: DependencyList = []
) {const [state, callback] = useAsyncFn(fn, deps, {loading: true,});useEffect(() => {callback();}, [callback]);return state;
}

解释

useAsync 这个 hook 用于处理异步操作,并且在 React 组件中管理它们的状态。我们逐步解析这个 API。

import { DependencyList, useEffect } from 'react';
import useAsyncFn from './useAsyncFn';
import { FunctionReturningPromise } from './misc/types';
//export type FunctionReturningPromise = (...args: any[]) => Promise<any>;
  • DependencyList: 这是 react 的一个类型,表示依赖项数组的类型,通常用于 useEffect 的第二个参数。
  • useEffect: React 的 hook,用于处理副作用。
  • useAsyncFn: 另一个自定义 hook,它处理异步函数的执行和状态管理。在下方会进行详细介绍。
  • FunctionReturningPromise: 一个 TypeScript 类型,表示返回 Promise 的函数类型。
export { AsyncState, AsyncFnReturn } from './useAsyncFn';

这行代码从 ./useAsyncFn 文件中导出了 AsyncStateAsyncFnReturn 类型或接口。它们通常用于描述异步操作的状态和结果。

函数 useAsync 实现:

  • 参数:

    • fn: T: 这是一个泛型参数,表示返回 Promise 的函数。这个函数将被执行并处理异步操作。
    • deps: DependencyList = []: 这是依赖项数组,默认为空数组,决定了 useEffect 何时重新执行。这个参数会传递给 useAsyncFn
  • 内部逻辑:

    • const [state, callback] = useAsyncFn(fn, deps, { loading: true });

      • useAsyncFn 是一个自定义 hook,用于处理异步操作。它返回一个包含状态和回调函数的数组。
      • state 是异步操作的状态(例如:loadingerrorresult)。
      • callback 是一个函数,用于触发异步操作。
    • useEffect(() => { callback(); }, [callback]);

      • 这个 useEffect 会在组件挂载时调用 callback。这样 fn 函数在组件加载时就会被执行一次。
      • 依赖项数组中包含 callback,意味着只有当 callback 改变时,useEffect 才会重新执行。
  • 返回值:

    • return state;
      • useAsync 返回的是 useAsyncFn 返回的 state,它包含异步操作的状态信息。

useAsyncFn

useAsync 就是基于 useAsyncFn 封装的,只是仅返回 state。

使用

import {useAsyncFn} from 'react-use';const Demo = ({url}) => {const [state, doFetch] = useAsyncFn(async () => {const response = await fetch(url);const result = await response.text();return result}, [url]);return (<div>{state.loading? <div>Loading...</div>: state.error? <div>Error: {state.error.message}</div>: <div>Value: {state.value}</div>}<button onClick={() => doFetch()}>Start loading</button></div>);
};

源码

import { DependencyList, useCallback, useRef, useState } from 'react';
import useMountedState from './useMountedState';
import { FunctionReturningPromise, PromiseType } from './misc/types';export type AsyncState<T> =| {loading: boolean;error?: undefined;value?: undefined;}| {loading: true;error?: Error | undefined;value?: T;}| {loading: false;error: Error;value?: undefined;}| {loading: false;error?: undefined;value: T;};type StateFromFunctionReturningPromise<T extends FunctionReturningPromise> = AsyncState<PromiseType<ReturnType<T>>
>;export type AsyncFnReturn<T extends FunctionReturningPromise = FunctionReturningPromise> = [StateFromFunctionReturningPromise<T>,T
];export default function useAsyncFn<T extends FunctionReturningPromise>(fn: T,deps: DependencyList = [],initialState: StateFromFunctionReturningPromise<T> = { loading: false }
): AsyncFnReturn<T> {const lastCallId = useRef(0);const isMounted = useMountedState();const [state, set] = useState<StateFromFunctionReturningPromise<T>>(initialState);const callback = useCallback((...args: Parameters<T>): ReturnType<T> => {const callId = ++lastCallId.current;if (!state.loading) {set((prevState) => ({ ...prevState, loading: true }));}return fn(...args).then((value) => {isMounted() && callId === lastCallId.current && set({ value, loading: false });return value;},(error) => {isMounted() && callId === lastCallId.current && set({ error, loading: false });return error;}) as ReturnType<T>;}, deps);return [state, callback as unknown as T];
}

解释

useAsyncFn 这个 hook 主要用于管理异步函数的状态,并提供一个函数来执行异步操作,同时保持组件状态的同步。我们来逐步解析这个 API。

  1. 类型定义
export type AsyncState<T> =| {loading: boolean;error?: undefined;value?: undefined;}| {loading: true;error?: Error | undefined;value?: T;}| {loading: false;error: Error;value?: undefined;}| {loading: false;error?: undefined;value: T;};
  • AsyncState<T>:
    • 这是一个 TypeScript 类型,表示异步操作的不同状态。它可以是:
      • 正在加载 (loading: true),没有值和错误。
      • 加载完成,有值 (loading: false, value: T)。
      • 加载完成,有错误 (loading: false, error: Error)。
      • 既不是加载中,也没有错误和值 (loading: falsevalue: T),或者只有错误。
//export type PromiseType<P extends Promise<any>> = P extends Promise<infer T> ? T : never;
//export type FunctionReturningPromise = (...args: any[]) => Promise<any>;
type StateFromFunctionReturningPromise<T extends FunctionReturningPromise> = AsyncState<PromiseType<ReturnType<T>>
>;
  • StateFromFunctionReturningPromise<T>:
    • 这个类型基于 AsyncStatePromiseType 生成,用于从返回 Promise 的函数中提取状态类型。即 useAsyncFn 接收的 fn 是一个动态的,其返回的数据类型是动态的,因此用 T 泛型来定义,并提取其对应的 AsyncState。
export type AsyncFnReturn<T extends FunctionReturningPromise = FunctionReturningPromise> = [StateFromFunctionReturningPromise<T>,T
];
  • AsyncFnReturn<T>:
    • 这是 useAsyncFn 返回的类型,包含两个元素:
      • StateFromFunctionReturningPromise<T>: 异步操作的状态。
      • T: 执行异步操作的函数。
  1. useAsyncFn 函数实现
  • 参数:

    • fn: T: 这是一个返回 Promise 的函数。
    • deps: DependencyList = []: 依赖项数组,默认为空数组,用于 useCallback
    • initialState: StateFromFunctionReturningPromise<T> = { loading: false }: 初始状态,默认为 { loading: false }
  • 内部逻辑:

    • const lastCallId = useRef(0);: 用于追踪最近一次异步调用的 ID,以便在异步操作完成时能正确更新状态。每次新增调用会 const callId = ++lastCallId.current;,而当 callId === lastCallId.current 为真才表示是本次异步返回值。

    • const isMounted = useMountedState();: 一个 hook 用于检查组件是否仍然挂载。这是为了避免在组件卸载后尝试更新状态。一般在异步调用后,需要更新变量时要结合该状态使用。

    • const [state, set] = useState<StateFromFunctionReturningPromise<T>>(initialState);: 组件的状态,用于存储异步操作的当前状态。

    • callback:

      • 使用 useCallback 包装的函数,会在组件挂载时执行异步操作。
      • 每次调用 callback 时,会生成一个新的 callId,并设置 loading 状态为 true
      • 调用 fn(...args),然后处理其结果:
        • 如果组件仍然挂载且 callId 匹配,则更新状态为操作的结果(成功或失败)。
  • 返回值:

    • [state, callback as unknown as T]: 返回一个数组,第一个元素是当前的异步状态,第二个元素是执行异步操作的函数。注意 callback 被强制转换为 T 类型,以便可以用作函数。

useAsyncRetry

使用

import {useAsyncRetry} from 'react-use';const Demo = ({url}) => {const state = useAsyncRetry(async () => {const response = await fetch(url);const result = await response.text();return result;}, [url]);return (<div>{state.loading? <div>Loading...</div>: state.error? <div>Error: {state.error.message}</div>: <div>Value: {state.value}</div>}{!loading && <button onClick={() => state.retry()}>Start loading</button>}</div>);
};

源码

import { DependencyList, useCallback, useState } from 'react';
import useAsync, { AsyncState } from './useAsync';export type AsyncStateRetry<T> = AsyncState<T> & {retry(): void;
};const useAsyncRetry = <T>(fn: () => Promise<T>, deps: DependencyList = []) => {const [attempt, setAttempt] = useState<number>(0);const state = useAsync(fn, [...deps, attempt]);const stateLoading = state.loading;const retry = useCallback(() => {if (stateLoading) {if (process.env.NODE_ENV === 'development') {console.log('You are calling useAsyncRetry hook retry() method while loading in progress, this is a no-op.');}return;}setAttempt((currentAttempt) => currentAttempt + 1);}, [...deps, stateLoading]);return { ...state, retry };
};export default useAsyncRetry;

解释

结合前面两个 api 的实现,比较简单,不做更多解释。

这篇关于WHAT - 通过 react-use 源码学习 React(Side-effects 篇)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Unity新手入门学习殿堂级知识详细讲解(图文)

《Unity新手入门学习殿堂级知识详细讲解(图文)》Unity是一款跨平台游戏引擎,支持2D/3D及VR/AR开发,核心功能模块包括图形、音频、物理等,通过可视化编辑器与脚本扩展实现开发,项目结构含A... 目录入门概述什么是 UnityUnity引擎基础认知编辑器核心操作Unity 编辑器项目模式分类工程

Python学习笔记之getattr和hasattr用法示例详解

《Python学习笔记之getattr和hasattr用法示例详解》在Python中,hasattr()、getattr()和setattr()是一组内置函数,用于对对象的属性进行操作和查询,这篇文章... 目录1.getattr用法详解1.1 基本作用1.2 示例1.3 原理2.hasattr用法详解2.

基于Python Playwright进行前端性能测试的脚本实现

《基于PythonPlaywright进行前端性能测试的脚本实现》在当今Web应用开发中,性能优化是提升用户体验的关键因素之一,本文将介绍如何使用Playwright构建一个自动化性能测试工具,希望... 目录引言工具概述整体架构核心实现解析1. 浏览器初始化2. 性能数据收集3. 资源分析4. 关键性能指

从入门到精通详解LangChain加载HTML内容的全攻略

《从入门到精通详解LangChain加载HTML内容的全攻略》这篇文章主要为大家详细介绍了如何用LangChain优雅地处理HTML内容,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录引言:当大语言模型遇见html一、HTML加载器为什么需要专门的HTML加载器核心加载器对比表二

前端如何通过nginx访问本地端口

《前端如何通过nginx访问本地端口》:本文主要介绍前端如何通过nginx访问本地端口的问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、nginx安装1、下载(1)下载地址(2)系统选择(3)版本选择2、安装部署(1)解压(2)配置文件修改(3)启动(4)

HTML中meta标签的常见使用案例(示例详解)

《HTML中meta标签的常见使用案例(示例详解)》HTMLmeta标签用于提供文档元数据,涵盖字符编码、SEO优化、社交媒体集成、移动设备适配、浏览器控制及安全隐私设置,优化页面显示与搜索引擎索引... 目录html中meta标签的常见使用案例一、基础功能二、搜索引擎优化(seo)三、社交媒体集成四、移动

HTML input 标签示例详解

《HTMLinput标签示例详解》input标签主要用于接收用户的输入,随type属性值的不同,变换其具体功能,本文通过实例图文并茂的形式给大家介绍HTMLinput标签,感兴趣的朋友一... 目录通用属性输入框单行文本输入框 text密码输入框 password数字输入框 number电子邮件输入编程框

HTML img标签和超链接标签详细介绍

《HTMLimg标签和超链接标签详细介绍》:本文主要介绍了HTML中img标签的使用,包括src属性(指定图片路径)、相对/绝对路径区别、alt替代文本、title提示、宽高控制及边框设置等,详细内容请阅读本文,希望能对你有所帮助... 目录img 标签src 属性alt 属性title 属性width/h

CSS3打造的现代交互式登录界面详细实现过程

《CSS3打造的现代交互式登录界面详细实现过程》本文介绍CSS3和jQuery在登录界面设计中的应用,涵盖动画、选择器、自定义字体及盒模型技术,提升界面美观与交互性,同时优化性能和可访问性,感兴趣的朋... 目录1. css3用户登录界面设计概述1.1 用户界面设计的重要性1.2 CSS3的新特性与优势1.

HTML5 中的<button>标签用法和特征

《HTML5中的<button>标签用法和特征》在HTML5中,button标签用于定义一个可点击的按钮,它是创建交互式网页的重要元素之一,本文将深入解析HTML5中的button标签,详细介绍其属... 目录引言<button> 标签的基本用法<button> 标签的属性typevaluedisabled