当前位置: 首页 > news >正文

React@16.x(24)自定义HOOK

目录

  • 1,介绍
  • 2,简单举例
    • 2.1,获取数据
    • 1.2,计时器
  • 2,自定义 HOOK 相比类组件

1,介绍

将一些常用的,跨组件的函数抽离,做成公共函数也就是 HOOK。自定义HOOK需要按照HOOK的规则来实现,才能像 useEffect 一样的使用。

可以看到,和项目中常用的工具方法 utils 类似,不同的是自定义HOOK 通常与业务强相关。

规则

  1. 函数名必须以 use 开头;
  2. 其他函数组件调用自定义HOOK函数时,应该放到顶层。

注意自定义HOOK 函数和函数组件的区别:要将函数当做组件来使用,函数名首字母必须大写

2,简单举例

2.1,获取数据

import { useEffect, useState } from "react";export const useList = () => {const [list, setList] = useState([]);useEffect(() => {(async () => {const res = await getFakerList();setList(res);})();}, []);return list;
};const getFakerList = () => {return new Promise((resolve) => {setTimeout(() => {resolve([1, 2, 3]);}, 1000);});
};

使用,

import React from "react";
import { useList } from "./hooks";
export default function App() {const list = useList();const li = list.map((m) => <li key={m}>{m}</li>);return <ul>{li}</ul>;
}

整体执行顺序:

  1. 函数组件调用 useList 函数
    1. 执行 useEffect 副作用函数,请求接口获取数据。
    2. 返回状态变量 list,此时获取的是空数组。

useEffect 副作用函数,使用立即执行函数的原因:
想使用 await 关键字,但 useEffect 的第一个参数有要求,如果有返回则必须是普通函数,所以不能把 async 关键字加到它上面。

type EffectCallback = () => (void | (() => void | undefined));
  1. 当模拟接口获取到数据,会更新状态变量 list,触发更新。所以 App.js 中会获取更新后的值渲染到页面。

可以将 useList 的函数体看做是在 App.js 中,也就是说,声明的状态其实放到了 App 函数组件中。
所以其实是 App 函数组件重新渲染,触发了 useList 函数重新调用。

1.2,计时器

import { useEffect } from "react";export const useTimer = (func, duration = 1000) => {useEffect(() => {const timer = setInterval(func, duration);return () => {clearInterval(timer);};}, []);
};

使用时,只需要关心要执行的内容即可。

import React from "react";
import { useTimer } from "./hooks";export default function App() {useTimer(() => {console.log("计时器的内容");});return <div>测试</div>;
}

2,自定义 HOOK 相比类组件

可以看到,通过HOOK实现的公共逻辑,如果在类组件中实现会比较繁琐,甚至会有到高阶组件才能实现。
而高阶组件虽然增强了类组件的功能,但带来了一些问题,比如 ref 转发,属性传递等,并且组件的嵌套层级也比较多,远不如 HOOK 来的方便。


以上。

http://www.lryc.cn/news/368209.html

相关文章:

  • 群体优化算法----树蛙优化算法介绍以及应用于资源分配示例
  • 常见汇编指令
  • Mysql学习(七)——约束
  • Redis实战篇02
  • 怎么用PHP语言实现远程控制两路照明开关
  • Docker面试整理-什么是多阶段构建?它的好处是什么?
  • ENSP校园网设计实验
  • 【Spring框架全系列】SpringBoot_3种配置文件_yml语法_多环境开发配置_配置文件分类(详细)
  • 华为坤灵路由器初始化的几个坑,含NAT配置
  • 【RAG入门教程04】Langchian的文档切分
  • 请求 响应
  • 技术周总结2024.06.03~06.09(K8S HikariCP数据库连接池)
  • 【JavaScript】了解 Sass:现代 CSS 的强大预处理器
  • 下载安装Thonny并烧录MicroPython固件至ESP32
  • YOLOv5改进 | 主干网络 | 将主干网络替换为轻量化的ShuffleNetv2【原理 + 完整代码】
  • LeetCode:字母异位词分组
  • 技术与业务的完美融合:大数据BI如何真正提升业务价值
  • 计网复习资料
  • 华为策略流控
  • 刷代码随想录有感(98):动态规划——爬楼梯
  • 零基础入门篇①⑦ Python可变序列类型--集合
  • 基于NodeJs 的Vue安装和创建项目
  • 【简单介绍下DALL-E2,什么是DALL-E2?】
  • springboot+mqtt使用总结
  • 搭建自己的组件库<2>dialog 组件
  • less学习笔记
  • 基于关键词自动采集抖音视频排名及互动数据(点赞、评论、收藏)
  • selenium中switch_to.window切换窗口的用法
  • 【nerf】nvidia-smi
  • 测试工具fio