vuejs3.0 从入门到精通——观察 Vue 实例从创建到销毁的完整生命周期

观察 Vue 实例从创建到销毁的完整生命周期

一、一个简单的 Vue 实例

代码如下:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8">
<title>Vue 从入门到精通,https://www.cnblogs.com/zuoyang/</title>
<script src="https://cdn.staticfile.org/vue/3.2.36/vue.global.min.js"></script>
</head>
<body>
<div id="example">
  <span>{{title}}</span>
</div>
 
<script>
const HelloVueApp = {
  data() {
    return {
      title: '你们的胃叫胃,孤的叫胃PLUS!'
    }
  }
}
 
Vue.createApp(HelloVueApp).mount('#example')
</script>
</body>
</html>

运行代码:

1
你们的胃叫胃,孤的叫胃PLUS!

二、生命周期

Vue 的实例在初始化时候,需要经历一系列的过程,如编译模版、渲染虚拟 DOM 树、将实例挂载到 DOM 上、设置数据监听和表数据绑定等。

在这些过程中,Vue 也允许开发者运行一些钩子函数,允许开发者在不同的阶段注入自己的代码。

举个 vue3 官方的例子,onMounted 钩子可以用来在组件完成初始渲染并创建 DOM 节点后运行代码:

1
2
3
4
5
6
7
<script setup>
import { onMounted } from 'vue'
 
onMounted(() => {
  console.log(`the component is now mounted.`)
})
</script>

还有其他一些钩子,会在实例生命周期的不同阶段被调用,最常用的是 onMountedonUpdated 和 onUnmounted。所有生命周期钩子的完整参考及其用法请参考 https://cn.vuejs.org/api/composition-api-lifecycle.html。

当调用 onMounted 时,Vue 会自动将回调函数注册到当前正被初始化的组件实例上。这意味着这些钩子应当在组件初始化时被同步注册。例如,请不要这样做:

1
2
3
4
5
6
setTimeout(() => {
  onMounted(() => {
    // 异步注册时当前组件实例已丢失
    // 这将不会正常工作
  })
}, 100)

注意这并不意味着对 onMounted 的调用必须放在 setup() 或 <script setup> 内的词法上下文中。onMounted() 也可以在一个外部函数中调用,只要调用栈是同步的,且最终起源自 setup() 就可以。

三、生命周期图示

四、生命周期钩子

4.1、onMounted()

说明:注册一个回调函数,在组件挂载完成后执行。

这个钩子在服务器端渲染期间不会被调用。

举例:通过模版引用访问一个元素:

1
2
3
4
5
6
7
8
9
10
11
12
13
<script setup>
import { ref, onMounted } from 'vue'
 
const el = ref()
 
onMounted(() => {
  el.value // <div>
})
</script>
 
<template>
  <div ref="el"></div>
</template>

4.2、onUpdated()

说明:注册一个回调函数,在组件因为响应式状态变更而更新其 DOM 树之后调用。

类型

1
function onUpdated(callback: () => void): void

  这个钩子在服务器端渲染期间不会被调用。

举例:访问更新后的 DOM:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
<script setup>
import { ref, onUpdated } from 'vue'
 
const count = ref(0)
 
onUpdated(() => {
  // 文本内容应该与当前的 `count.value` 一致
  console.log(document.getElementById('count').textContent)
})
</script>
 
<template>
  <button id="count" @click="count++">{{ count }}</button>
</template>

4.3、onUnmounted()

说明:注册一个回调函数,在组件实例被卸载之后调用。

类型:

1
function onUnmounted(callback: () => void): void

这个钩子在服务器端渲染期间不会被调用。

举例:

1
2
3
4
5
6
7
8
9
10
11
12
<script setup>
import { onMounted, onUnmounted } from 'vue'
 
let intervalId
onMounted(() => {
  intervalId = setInterval(() => {
    // ...
  })
})
 
onUnmounted(() => clearInterval(intervalId))
</script>

4.4、onBeforeMount()

说明:注册一个钩子,在 组件被挂载之前被调用。

类型:

1
function onBeforeMount(callback: () => void): void

  详细说明:

当这个钩子被调用时 ,组件已完成了其响应状态的设置,但还没有创建 DOM 节点。它即将首次执行 DOM 渲染过程。

这个钩子在服务器端渲染期间不会被调用。

4.5、onBeforeUpdate()

说明:注册一个钩子,在组件即将因为响应式状态变更而更新其 DOM 树之前调用。

类型:

1
function onBeforeUpdate(callback: () => void): void

详细说明:

这个钩子可以用来在 Vue 更新 DOM 之前访问 DOM 状态。在这个钩子中更改状态也是安全的。

这个钩子在服务器端渲染期间不会被调用。

4.6、onBeforeUnmount()

说明:注册一个钩子,在组件实例被卸载之前调用。

类型:

1
function onBeforeUnmount(callback: () => void): void

详细信息:

当这个钩子被调用时,组件实例依然会保有全部的功能。

这个钩子在服务器端渲染期间不会被调用。

4.7、onErrorCaptured()

说明:注册一个钩子,在捕获了后端组件传递的错误时调用。

类型:

1
2
3
4
5
6
7
function onErrorCaptured(callback: ErrorCapturedHook): void
 
type ErrorCapturedHook = (
  err: unknown,
  instance: ComponentPublicInstance | null,
  info: string
) => boolean | voidx

详细信息:

错误可以从以下的几个来源中捕获:

        • 组件渲染
        • 事件处理器
        • 生命周期钩子
        • setup() 函数
        • 侦听器
        • 自定义指令钩子
        • 过渡钩子

这个钩子有三个实参:

      • 错误对象
      • 触发改错误的组件实例
      • 说明错误来源类型的信息字符串

错误传递规则:

        • 默认情况下,所有的错误都会被发送到应用级的 app.config.errorHandler (前提是这个函数已经定义),这样这些错误都能在一个统一的地方报告给分析服务。

        • 如果组件的继承链或组件链上存在多个 errorCaptured 钩子,对于同一个错误,这些钩子会被按从底至上的顺序一一调用。这个过程被称为“向上传递”,类似于原生 DOM 事件的冒泡机制。

        • 如果 errorCaptured 钩子本身抛出了一个错误,那么这个错误和原来捕获到的错误都将被发送到 app.config.errorHandler

        • errorCaptured 钩子可以通过返回 false 来阻止错误继续向上传递。即表示“这个错误已经被处理了,应当被忽略”,它将阻止其他的 errorCaptured 钩子或 app.config.errorHandler 因这个错误而被调用。

4.8、onRenderTracked()

说明:注册一个调试钩子,当组件渲染过程中追踪到响应式依赖时调用。

类型:

1
2
3
4
5
6
7
8
9
10
function onRenderTracked(callback: DebuggerHook): void
 
type DebuggerHook = (e: DebuggerEvent) => void
 
type DebuggerEvent = {
  effect: ReactiveEffect
  target: object
  type: TrackOpTypes /* 'get' | 'has' | 'iterate' */
  key: any
}

这个钩子仅在开发模式下可用,且在服务器端渲染期间不会被调用。

4.9、onRenderTriggered()

说明:注册一个调试钩子,当响应式依赖的变更触发了组件渲染时调用。

类型:

1
2
3
4
5
6
7
8
9
10
11
12
13
function onRenderTriggered(callback: DebuggerHook): void
 
type DebuggerHook = (e: DebuggerEvent) => void
 
type DebuggerEvent = {
  effect: ReactiveEffect
  target: object
  type: TriggerOpTypes /* 'set' | 'add' | 'delete' | 'clear' */
  key: any
  newValue?: any
  oldValue?: any
  oldTarget?: Map<any, any> | Set<any>
}

这个钩子仅在开发模式下可用,且在服务器端渲染期间不会被调用。

4.10、onActivated()

说明:注册一个回调函数,若组件实例是 <KeepAlive> 缓存树的一部分,当组件被插入到 DOM 中时调用。

类型: 

1
function onActivated(callback: () => void): void

这个钩子在服务器端渲染期间不会被调用。

4.11、onDeactivated()

说明:注册一个回调函数,若组件实例是 <KeepAlive> 缓存树的一部分,当组件从 DOM 中被移除时调用。

类型:

1
function onDeactivated(callback: () => void): void

这个钩子在服务器端渲染期间不会被调用。

4.12、onServerPrefetch()

说明:注册一个异步函数,在组件实例在服务器上被渲染之前调用。

类型:  

1
function onServerPrefetch(callback: () => Promise<any>): void

详细信息:

  • 如果这个钩子返回了一个 Promise,服务端渲染会在渲染该组件前等待该 Promise 完成。
  • 这个钩子仅会在服务端渲染中执行,可以用于执行一些仅存在于服务端的数据抓取过程。 
举例:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
<script setup>
import { ref, onServerPrefetch, onMounted } from 'vue'
 
const data = ref(null)
 
onServerPrefetch(async () => {
  // 组件作为初始请求的一部分被渲染
  // 在服务器上预抓取数据,因为它比在客户端上更快。
  data.value = await fetchOnServer(/* ... */)
})
 
onMounted(async () => {
  if (!data.value) {
    // 如果数据在挂载时为空值,这意味着该组件
    // 是在客户端动态渲染的。将转而执行
    // 另一个客户端侧的抓取请求
    data.value = await fetchOnClient(/* ... */)
  }
})
</script> 
posted @   左扬  阅读(265)  评论(0编辑  收藏  举报
相关博文:
阅读排行:
· 阿里最新开源QwQ-32B,效果媲美deepseek-r1满血版,部署成本又又又降低了!
· 开源Multi-agent AI智能体框架aevatar.ai,欢迎大家贡献代码
· Manus重磅发布:全球首款通用AI代理技术深度解析与实战指南
· 被坑几百块钱后,我竟然真的恢复了删除的微信聊天记录!
· AI技术革命,工作效率10个最佳AI工具
levels of contents
点击右上角即可分享
微信分享提示