【vue高频面试题】第 23 题:Vue3 自定义指令(directive)完整解析

第 23 题:Vue3 自定义指令(directive)完整解析


🎯 一、什么是自定义指令?

自定义指令用于在DOM 层 封装可复用的逻辑,

适合处理:聚焦、懒加载、权限、拖拽、固定吸顶、长按 等场景。

Vue 提供两个层级:

  • 全局指令(app.directive)
  • 组件局部指令(directives: { xxx })

用途与组件不同:

  • 组件 → 处理 UI 和业务逻辑
  • 指令 → 只负责 DOM 操作

🎯 二、Vue3 自定义指令生命周期(重点)

Vue3 中指令共有 5 个钩子:

  1. created(el, binding)
  2. beforeMount
  3. mounted
  4. beforeUpdate
  5. updated
  6. beforeUnmount
  7. unmounted

最常用:mountedupdated


🎯 三、最简单示例:v-focus(输入框自动聚焦)

局部指令

xml 复制代码
<input v-focus />

<script>
export default {
  directives: {
    focus: {
      mounted(el) {
        el.focus()
      }
    }
  }
}
</script>

🎯 四、Vue3 自定义指令原理(高频面试点)

Vue 在编译模板时,会将所有指令解析为 指令对象,在渲染 DOM 时调用相应的指令钩子,对 DOM 进行增强。

执行流程:

  1. 模板编译成 VNode 时指令信息会被记录
  2. 渲染时调用 mounted(el, binding)
  3. 更新时调用 updated(el, binding)
  4. 销毁时调用 unmounted(el)

本质是:

VNode 生命周期 + 指令钩子 = DOM 增强


🎯 五、常见指令面试案例(重要)

1️⃣ 图片懒加载 v-lazy

scss 复制代码
mounted(el, binding) {
  const observer = new IntersectionObserver(([entry]) => {
    if (entry.isIntersecting) {
      el.src = binding.value
      observer.unobserve(el)
    }
  })
  observer.observe(el)
}

技术点:使用 IntersectionObserver 监听是否进入视口。


2️⃣ 权限指令 v-permission

scss 复制代码
mounted(el, binding) {
  const userRoles = store.state.user.roles
  if (!userRoles.includes(binding.value)) {
    el.remove() // 无权限则移除节点
  }
}

很多后台系统都会问这个。


3️⃣ 长按指令 v-longpress

javascript 复制代码
mounted(el, binding) {
  let timer
  el.addEventListener("mousedown", () => {
    timer = setTimeout(() => binding.value(), 500)
  })
  el.addEventListener("mouseup", () => clearTimeout(timer))
}

🎯 六、binding 参数(面试必问)

binding 包含:

arduino 复制代码
binding = {
  instance,           // 组件实例
  value,              // 传入的值
  oldValue,           // 旧值
  arg,                // 指令参数
  modifiers           // 修饰符
}

用法示例:

ini 复制代码
<div v-demo:color.red="msg"></div>

解析:

  • arg = 'color'
  • modifiers = { red: true }
  • value = msg

🎯 七、关键知识点 - 什么时候用指令?什么时候用组件?

用组件? 用指令?
有 UI 和模板结构 不需要模板结构
需要逻辑复用 + DOM 只需要封装 DOM 操作
有状态 无状态逻辑(纯 DOM)
适合全局组件库 适合权限、懒加载、拖拽

一句话:

只要是"DOM 行为",尽量用指令;
数据逻辑 + UI 就应该用组件。


🎯 八、面试官常问追问(附最佳答案)


追问 1:Vue3 的指令和 Vue2 有哪些不同?

答:

  • Vue3 去掉了 bindinserted
  • 新增了 createdbeforeMount
  • 生命周期更加统一
  • binding.instance 替代 vnode.component

追问 2:指令能在 setup 中使用吗?

不能直接在 setup return 中注册,但可以写成一个对象:

javascript 复制代码
const vFocus = {
  mounted(el) { el.focus() }
}

return { vFocus }

追问 3:指令和 watch 哪个性能更好?

使用指令,因为:

  • watch 会触发计算
  • 指令只在 DOM 层操作,不依赖响应式系统
  • 性能更高

追问 4:指令能否访问组件内部变量?

可以,通过 binding.instance

复制代码
binding.instance.xxx

🎯 九、一句话总结

Vue3 自定义指令是操作 DOM 的利器,不参与组件逻辑,拥有独立生命周期,常用于懒加载、权限、拖拽、聚焦等场景。

相关推荐
小码哥_常2 分钟前
Android消息机制:Handler、Looper和Message的深度剖析
前端
小码哥_常4 分钟前
安卓开发新姿势:文件Picker全攻略,无痛适配不再难
前端
happymaker062615 分钟前
web前端学习日记——DAY04
前端·学习
发现一只大呆瓜21 分钟前
React-路由监听 / 跳转 / 守卫全攻略(附实战代码)
前端·react.js·面试
swipe1 小时前
为什么 RAG 一定离不开向量检索:从文档向量化到语义搜索的工程实现
前端·llm·agent
OpenTiny社区2 小时前
AI-Extension:让 AI 真的「看得到、动得了」你的浏览器
前端·ai编程·mcp
IT_陈寒2 小时前
Redis缓存击穿:3个鲜为人知的防御策略,90%开发者都忽略了!
前端·人工智能·后端
农夫山泉不太甜3 小时前
Tauri v2 实战代码示例
前端
消失的旧时光-19433 小时前
Android 面试高频:JSON 文件、大数据存储与断电安全(从原理到工程实践)
android·面试·json
yuhaiqiang3 小时前
被 AI 忽悠后,开始怀念搜索引擎了?
前端·后端·面试