Vue3核心语法进阶(Hook)

Vue3 自定义 Hook:让你的代码像乐高一样“可复用”!

大家好,我是你们的前端小伙伴!上一篇我们聊了 Vue3 的生命周期,今天咱们继续深入 Vue3 的核心利器——自定义 Hook(Custom Hook)

如果你已经用过 refreactivewatchonMounted 这些 Composition API,那你已经“会用工具了”。而今天我们要学的,是如何把一堆工具打包成一个“超级工具包”,哪里需要就往哪里一放,功能立马就有了!

这个“超级工具包”,就是 自定义 Hook


什么是自定义 Hook?

想象一下,你在写一个电商网站,有 5 个页面都需要“获取用户信息 + 检查登录状态 + 显示用户头像”。

如果你在每个页面都重复写一遍:

const userInfo = ref(null)
const isLoggedIn = ref(false)onMounted(async () => {const data = await fetch('/api/user')userInfo.value = dataisLoggedIn.value = data ? true : false
})watch(userInfo, (newVal) => {if (newVal) console.log('用户已登录')
})

那……恭喜你,代码已经“复制粘贴”了 5 遍!如果哪天接口变了,你得改 5 个地方,想想都头大。

自定义 Hook 就是:把这段“通用逻辑”抽出来,变成一个函数,谁需要谁调用!


自定义 Hook 的本质

它就是一个以 use 开头的普通函数,比如 useUseruseCartuseLocalStorage

它内部可以使用任何 Composition API(refwatchonMounted 等),然后把需要暴露的数据和方法 return 出去。


动手写一个:useUser() 用户信息 Hook

我们来写一个真正可用的自定义 Hook!

// composables/useUser.js
import { ref, onMounted, watch } from 'vue'export function useUser() {const userInfo = ref(null)const isLoggedIn = ref(false)const loading = ref(true)// 模拟请求用户数据const fetchUser = async () => {loading.value = truetry {const res = await fetch('/api/user')const data = await res.json()userInfo.value = dataisLoggedIn.value = !!data} catch (err) {console.error('获取用户失败', err)} finally {loading.value = false}}// 组件挂载时自动获取用户onMounted(() => {fetchUser()})// 监听用户信息变化watch(userInfo, (newVal) => {if (newVal) {console.log(`欢迎回来,${newVal.name}!`)}})// 把你需要的东西 return 出去return {userInfo,isLoggedIn,loading,fetchUser // 也可以手动刷新}
}

文件名建议:composables/useXxx.js,这是 Vue 社区的约定。


在组件中使用它

现在,任何组件只要想用“用户逻辑”,一句话搞定:

<!-- UserProfile.vue -->
<script setup>
import { useUser } from '@/composables/useUser'// 一行代码,搞定用户逻辑!
const { userInfo, isLoggedIn, loading, fetchUser } = useUser()
</script><template><div v-if="loading">加载中...</div><div v-else-if="isLoggedIn"><h2>欢迎,{{ userInfo.name }}!</h2><button @click="fetchUser">刷新</button></div><div v-else>请先登录</div>
</template>

看到了吗?组件代码变得极其干净,逻辑都被“封装”到 useUser 里了。


自定义 Hook 的核心优势

优势说明
逻辑复用一套登录逻辑,10 个页面都能用
解耦清晰组件只管“视图”,Hook 管“逻辑”
易于测试直接测试 useUser 函数,不用渲染组件
类型友好TypeScript 能自动推导返回值类型

高级用法:支持参数和返回函数

自定义 Hook 不只是“固定逻辑”,它也可以很灵活!

示例:useLocalStorage —— 让数据自动存到本地

// composables/useLocalStorage.js
import { ref, watch } from 'vue'export function useLocalStorage(key, initialValue) {// 从 localStorage 读取,或用默认值const data = ref(JSON.parse(localStorage.getItem(key)) || initialValue)// 数据变化时,自动存到 localStoragewatch(data, (newVal) => {localStorage.setItem(key, JSON.stringify(newVal))}, { deep: true }) // deep: true 支持对象/数组return data
}

使用它:

// 计数器,刷新页面也不会丢!
const count = useLocalStorage('count', 0)// 用户设置
const settings = useLocalStorage('userSettings', { theme: 'dark' })

看,一个 Hook,支持任意 key 和默认值,真正做到了“通用”


常见误区 & 注意事项

误区 1:在普通函数里用 ref 就是 Hook?

No!只有在 setup<script setup> 中调用的函数,才能使用 Composition API。你的 useXxx 函数必须在组件上下文中调用。

误区 2:Hook 可以 return 模板?

不能!Hook 只负责逻辑和数据,模板还是得在组件里写。

最佳实践:命名规范

  • 一定要以 use 开头,比如 useMouseuseScrolluseFetch
  • 返回值尽量结构清晰,方便解构使用
  • 复杂逻辑可以拆分成多个小 Hook

实战案例:useMouse —— 跟踪鼠标位置

// composables/useMouse.js
import { ref, onMounted, onUnmounted } from 'vue'export function useMouse() {const x = ref(0)const y = ref(0)const update = (e) => {x.value = e.clientXy.value = e.clientY}onMounted(() => {window.addEventListener('mousemove', update)})onUnmounted(() => {window.removeEventListener('mousemove', update)})return { x, y }
}

使用:

<script setup>
import { useMouse } from '@/composables/useMouse'
const { x, y } = useMouse()
</script><template><div>鼠标位置:{{ x }}, {{ y }}</div>
</template>

瞬间,你的组件就有了“追踪鼠标”的超能力!


为什么叫“Hook”?

“Hook” 的意思是“钩子”,它就像是把一段逻辑“钩”进组件的生命周期中。

比如 onMounted 就是一个“钩子”,告诉 Vue:“等组件挂载后,执行我这个函数”。

而自定义 Hook,就是把多个钩子和逻辑打包,变成一个可复用的“增强包”。


总结:自定义 Hook 的灵魂

关键点说明
目的逻辑复用,避免重复代码
形式useXxx() 函数,返回响应式数据和方法
能力可接收参数、可组合多个 API、可嵌套使用
好处代码更清晰、维护更容易、团队协作更高效

最后一句话

组件负责“长什么样”,Hook 负责“怎么动”。

把通用逻辑封装成 Hook,你的代码就会像乐高积木一样,拼装自由,复用无忧

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如若转载,请注明出处:http://www.pswp.cn/web/91947.shtml
繁体地址,请注明出处:http://hk.pswp.cn/web/91947.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

工控领域协议之Modbus

Modbus 是一种通信协议&#xff0c;用于工业自动化领域中的设备之间的通信。它是一种串行通信协议&#xff0c;广泛应用于连接不同设备、传感器和执行器的工业控制系统。 Modbus 在工业控制系统、自动化设备、能源管理系统等领域得到广泛应用。 Modbus 协议的基本特点&#xff…

大件垃圾识别 mAP↑28%:陌讯多模态融合算法实战解析

一、行业痛点&#xff1a;大件垃圾识别的现实困境在城市环卫智能化转型过程中&#xff0c;大件垃圾&#xff08;如废旧家具、电器等&#xff09;的自动化识别与分拣成为关键环节。据住建部《城市环境卫生发展报告》显示&#xff0c;传统人工分拣模式下大件垃圾识别准确率不足 6…

vk框架或者普通函数封装的一些函数可以拿取使用【会持续更新】

1.身份证校验【通用】/*** function isIDCard* description 判断是否为有效的身份证号码。* param {string} idCard - 待验证的身份证号码。* returns {boolean} 返回验证结果。*/ pubFun.isIDCard function (idCard) {// 身份证号码为15位或者18位&#xff0c;15位时全为数字…

如何给Word和WPS文档添加密码或取消密码

要保护Word和WPS文档&#xff0c;可以为它们加密&#xff0c;加密有两类&#xff1a;打开密码和修改密码。密码设置有两个入口&#xff0c;一个是在另存为&#xff0c;一个是在文件菜单。Word和WPS文字的路径略有不同&#xff0c;微软Office和WPS的其他套件也是如此操作。一、W…

uni-app项目gitignore文件示例

uni-app 忽略以下文件和目录 DS_Store 忽略 UniApp 编译生成的小程序相关目录 unpackage/ uni_modules/ 忽略编辑器自动生成的文件 idea/ vscode/ 忽略日志文件 logs/ 忽略临时文件 temp/ 忽略构建工具自动生成的文件 build/ 忽略 npm 安装的包文件 package-lock.json yarn.loc…

LeetCode 135:分糖果

LeetCode 135&#xff1a;分糖果问题本质与核心挑战 给定孩子的评分数组&#xff0c;需满足 “每个孩子至少1颗糖果&#xff0c;相邻评分高的孩子糖果更多”&#xff0c;求最少糖果总数。核心挑战&#xff1a; 相邻约束是双向的&#xff08;左→右和右→左都需满足&#xff09;…

【QT】安装与配置

个人主页&#xff1a;Guiat 归属专栏&#xff1a;QT 文章目录1. QT简介与准备工作1.1 什么是QT1.2 QT的版本选择1.3 系统要求检查2. QT安装方式详解2.1 官方在线安装器2.2 离线安装包2.3 包管理器安装3. Windows平台安装配置3.1 Windows安装步骤3.2 环境变量配置3.3 Visual Stu…

Java从入门到精通 - 算法、正则、异常

算法、正则、异常 此笔记参考黑马教程&#xff0c;仅学习使用&#xff0c;如有侵权&#xff0c;联系必删 文章目录算法、正则、异常1. 常见算法1.1 简单认识算法1.1.1 什么是算法&#xff1f;1.1.2 为什么要学习算法&#xff1f;1.2 排序算法1.2.1 冒泡排序1.2.1.1 实现冒泡排…

题单【排序】

P1271 【深基9.例1】选举学生会 P1271 【深基9.例1】选举学生会 - 洛谷 【方法一】快速排序 使用sort()&#xff0c;注意数组的范围&#xff01;&#xff01;&#xff01; #include<bits/stdc.h> using namespace std;int a[2000000],n,m;int main() {cin>>n>&g…

【机器学习】(算法优化二)提升算法之:AdaBoost与随机梯度

文章目录一、 AdaBoost&#xff1a;自适应提升算法1、AdaBoost数学原理详解1.1、 目标函数1.2、 样本权重更新的逻辑1.3、 模型权重计算的含义1.4、 AdaBoost的核心思想2、为什么AdaBoost如此有效&#xff1f;二、 随机梯度提升算法&#xff1a;梯度优化下更精细的优化1、随机梯…

力扣 hot100 Day65

75. 颜色分类 给定一个包含红色、白色和蓝色、共 n 个元素的数组 nums &#xff0c;原地 对它们进行排序&#xff0c;使得相同颜色的元素相邻&#xff0c;并按照红色、白色、蓝色顺序排列。 我们使用整数 0、 1 和 2 分别表示红色、白色和蓝色。 必须在不使用库内置的 sort 函…

12.Linux 磁盘管理

Linux : 磁盘管理 一、磁盘设备命名规则磁盘类型设备命名模式示例特点SATA/SCSI/SAS/dev/sdXsda&#xff08;第一块硬盘&#xff09; sda1&#xff08;第一块硬盘第一分区&#xff09;机械硬盘/通用接口NVMe/dev/nvmeXnYpZnvme0n1&#xff08;第一通道第一块盘&#xff09; …

《Linux服务与安全管理》| DHCP服务器安装和配置

《Linux服务与安全管理》| DHCP服务器安装和配置 目录 《Linux服务与安全管理》| DHCP服务器安装和配置 一、点击“编辑虚拟机设置”&#xff0c;配置三台虚拟机为“仅主机”模式。 二、server01开机&#xff0c;root用户登录&#xff0c;输入nmtui&#xff0c;进入图形界面…

赛博威携手Dify,助力AI在企业的场景化落地

人工智能正以前所未有的速度重塑商业世界。我们经历了从理论探索到大语言模型&#xff08;LLM&#xff09;的爆发式增长&#xff0c;如今&#xff0c;一个以“AI Agent&#xff08;智能体&#xff09;”为核心的新阶段已然来临。AI Agent代表了人工智能应用的未来形态。它不再被…

嵌入式硬件中三极管推挽电路控制与实现

我们昨天讲到了这个电路。 如果 A 电是 PWM 波,那么请问 B 点是不是 PWM 波呢?那么,当 PWM 为高时, B 点的电流是从哪里流过来的?

数据结构——查找(三、树形查找)

一、二叉排序树&#xff08;BST&#xff09;1、二叉排序树的定义构造一棵二叉排序树的目的并不是排序&#xff0c;而是提高查找、插入和删除关键字的速度二叉排序树&#xff08;也称二叉搜索树&#xff09;或者是一颗空树&#xff0c;或者是具有以下性质的二叉树1、若左子树非空…

八股——Kafka相关

文章目录1、 消息队列的作用什么&#xff1f;思&#xff1a;消息队列是什么?消息队列的定义消息队列的工作原理消息队列的作用消息队列的常见类型消息队列的简单例子2、Kafka 集群的架构是什么样子的&#xff1f;3、Kafka 消费者组和生产者组是什么&#xff1f;定义与核心作用…

墨者学院SQL手工注入漏洞测试(MySQL数据库)题目,纯手工注入教程

打开练习手工注入的靶场,发现此时为一个登录页面,我们先试着登录看看注入点在不在登录页面 使用用户:or 1=1# 密码:admin123;尝试登录,发现显示错误后直接弹回原页面,无sql报错相关语句,这里不存在sql注入点 一:判断注入点以及猜测是否有注入 此时点击这里的动态页面…

[硬件电路-140]:模拟电路 - 信号处理电路 - 锁定放大器概述、工作原理、常见芯片、管脚定义

一、锁定放大器概述锁定放大器&#xff08;Lock-in Amplifier&#xff09;是一种基于相干检测技术的高灵敏度测量仪器&#xff0c;通过将待测信号与参考信号进行同步处理&#xff0c;从强噪声中提取微弱信号并精确测量其振幅与相位。其核心优势包括&#xff1a;信噪比提升&…

下载 | Windows Server 2025官方原版ISO映像!(7月更新、标准版、数据中心版、26100.4652)

⏩ 资源A066_Windows_Server_2025系统映像&#x1f536; Windows Server 2025官方原版ISO映像&#xff0c;7月更新版已放出。提供来自微软官方每月更新的ISO原版映像&#xff0c;内部包含了标准版和数据中心版&#xff0c;可选择无GUI界面版或桌面体验版&#xff0c;满足不同部…