入门 - vue整个过程的生命周期详解

news/2024/12/22 9:05:09/

生命周期概念

Vue的生命周期就是vue实例从创建到销毁的全过程,也就是new Vue()开始就是vue生命周期的开始。Vue 实例有⼀个完整的⽣命周期,也就是从开始创建、初始化数据、编译模版、挂载Dom->渲染、更新->渲染、卸载 等⼀系列过程,称这是Vue的⽣命周期。钩子函数是Vue生命周期中每个阶段对外开放让程序员操作Vue的接口。Vue有8个钩子函数。

每个vue实例从创建到销毁的过程都是一个生命周期,也会运行对应的钩子函数,下图为Vue生命周期示意图:

8个钩子函数

1. beforeCreate

官方解释: 在实例初始化之后,数据观测(data observer)和event/watcher事件配置之前被调用。

说明:这个时候this还不能使用,data中的数据、methods中的方法,以及watcher中的事件都不能获得。

var vm = new Vue({
el: '#app',
data: {message: '今天是周五!!!'
},
beforeCreate(){console.group('beforeCreate 创建前状态==========>>');console.log("%c%s", "color:red", "el     : "+this.$el);   //undefinedconsole.log("%c%s", "color:red", "data   : "+this.$data); //undefinedconsole.log("%c%s", "color:red", "message: "+this.message);   //undefined
},
//...

2. created

官方解释: 实例已经创建完成之后被调用。在这一步,实例已完成以下的配置:数据观测(data observer),属性和方法的运算,watch/event 事件回调。然而,挂载阶段还没开始,$el属性目前不可见。

说明:这个时候可以操作vue中的数据和方法,但是还不能对dom节点进行操作。

//...
created(){console.group('created 创建完毕状态==========>>');console.log("%c%s", "color:red", "el     : "+this.$el);   //undefinedconsole.log("%c%s", "color:red", "data   : "+this.$data); //[object Object]console.log("%c%s", "color:red", "message: "+this.message);   //今天是周五!!!
},

3. beforeMount

官方解释: 在挂载开始之前被调用:相关的render函数首次被调用。

说明:$el属性已存在,是虚拟dom,只是数据未挂载到模板中。

//...
beforeMount(){console.group('beforeMount 挂载前状态==========>>');console.log("%c%s", "color:red", "el     : "+this.$el);   //[object HTMLDivElement]console.log(this.$el);console.log("%c%s", "color:red", "data   : "+this.$data); //[object Object]console.log("%c%s", "color:red", "message: "+this.message);   //今天是周五!!!
},

4. mounted

官方解释: el被新创建的vm.$el替换,并挂载到实例上去之后调用该钩子。如果 root 实例挂载了一个文档内元素,当mounted被调用时vm.$el也在文档内。

注意mounted不会承诺所有的子组件也都一起被挂载。如果你希望等到整个视图都渲染完毕,可以用vm.$nextTick替换掉mounted

说明:挂载完毕,这时dom节点被渲染到文档内,dom操作在此时能正常进行

//...
mounted(){
console.group('mounted 挂载结束状态==========>>');
console.log("%c%s", "color:red", "el     : "+this.$el);   //[object HTMLDivElement]
console.log(this.$el);  
console.log("%c%s", "color:red", "data   : "+this.$data); //[object Object]
console.log("%c%s", "color:red", "message: "+this.message);   //今天是周五!!!
},
//...

点击页面中的元素执行相应的事件,并触发beforeUpdate和updated钩子函数。

5. beforeUpdate

官方解释: 数据更新时调用,发生在虚拟 DOM 打补丁之前。这里适合在更新之前访问现有的 DOM,比如手动移除已添加的事件监听器。

说明:beforeUpdate是指view层的数据变化前,不是data中的数据改变前触发。因为Vue是数据驱动的。

//...
beforeUpdate(){
console.group('beforeUpdate 更新前状态==========>>');
console.log("%c%s", "color:red", "el     : "+this.$el);   //[object HTMLDivElement]
console.log(this.$el);
console.log(this.$el.innerHTML);    //<p>今天是周一!!!</p>
console.log("%c%s", "color:red", "data   : "+this.$data); //[object Object]
console.log("%c%s", "color:red", "message: "+this.message);   //今天周二了!!!
},
//...

6. updated

官方解释: 由于数据更改导致的虚拟DOM重新渲染和打补丁,在这之后会调用该钩子。

当这个钩子被调用时,组件DOM已经更新,所以你现在可以执行依赖于DOM的操作。然而在大多数情况下,你应该避免在此期间更改状态。如果要相应状态改变,通常最好使用计算属性或watcher取而代之。

注意updated不会承诺所有的子组件也都一起被重绘。如果你希望等到整个视图都重绘完毕,可以用vm.$nextTick替换掉 updated:

说明:view层的数据更新后,data中的数据同beforeUpdate,都是更新完以后的。

//...
updated(){
console.group('updated 更新完成状态==========>>');
console.log("%c%s", "color:red", "el     : "+this.$el);   //[object HTMLDivElement]
console.log(this.$el);
console.log(this.$el.innerHTML);    //<p>今天周二了!!!</p>
console.log("%c%s", "color:red", "data   : "+this.$data); //[object Object]
console.log("%c%s", "color:red", "message: "+this.message);   //今天周二了!!!
},
//...

注意:细心的小伙伴会发现beforeUpdate和updated钩子函数中的$el一样,根据官方理解beforeUpdate应该指向虚拟dom,所以才会一样,而dom中的真正内容不一样,但是beforeMount和mouted钩子函数中为什么又会有区别呢?感觉是设计的不足之处。

执行vm.$destroy()函数触发beforeDestroy和destoryed钩子函数

7. beforeDestroy

官方解释: 实例销毁之前调用。在这一步,实例仍然完全可用。

说明:

//...
beforeDestroy(){
console.group('beforeDestroy 销毁前状态==========>>');
console.log("%c%s", "color:red", "el     : "+this.$el);   //[object HTMLDivElement]
console.log(this.$el);
console.log("%c%s", "color:red", "data   : "+this.$data); //[object Object]
console.log("%c%s", "color:red", "message: "+this.message);   //今天周二了!!!
},
//...

8. destroyed

官方解释: Vue实例销毁后调用。调用后,Vue实例指示的所有东西都会解绑定,所有的事件监听器会被移除,所有的子实例也会被销毁。

说明:执行destroy方法后,对data的改变不会再触发周期函数,此时的vue实例已经解除了事件监听以及和dom的绑定,但是dom结构依然存在。

//...
destroyed(){
console.group('destroyed 销毁完成状态==========>>');
console.log("%c%s", "color:red", "el     : "+this.$el);   //[object HTMLDivElement]
console.log(this.$el);
console.log("%c%s", "color:red", "data   : "+this.$data); //[object Object]
console.log("%c%s", "color:red", "message: "+this.message);   //今天周二了!!!
},
//...

代码如下:

<!DOCTYPE html>
<html>
<head><title></title>
</head>
<body>
<div id="app" @click="change"><p>{{message}}</p>
</div>
<script src="https://cdn.bootcss.com/vue/2.6.10/vue.min.js"></script>
<script type="text/javascript">
var vm = new Vue({el: '#app',data: {message: '今天是周五!!!'},beforeCreate(){console.group('beforeCreate 创建前状态==========>>');console.log("%c%s", "color:red", "el     : "+this.$el);   //undefinedconsole.log("%c%s", "color:red", "data   : "+this.$data); //undefinedconsole.log("%c%s", "color:red", "message: "+this.message);   //undefined},created(){console.group('created 创建完毕状态==========>>');console.log("%c%s", "color:red", "el     : "+this.$el);   //undefinedconsole.log("%c%s", "color:red", "data   : "+this.$data); //[object Object]console.log("%c%s", "color:red", "message: "+this.message);   //今天是周六!!!},beforeMount(){console.group('beforeMount 挂载前状态==========>>');console.log("%c%s", "color:red", "el     : "+this.$el);   //[object HTMLDivElement]console.log(this.$el);console.log("%c%s", "color:red", "data   : "+this.$data); //[object Object]console.log("%c%s", "color:red", "message: "+this.message);   //今天是周六!!!},mounted(){console.group('mounted 挂载结束状态==========>>');console.log("%c%s", "color:red", "el     : "+this.$el);   //[object HTMLDivElement]console.log(this.$el);  console.log("%c%s", "color:red", "data   : "+this.$data); //[object Object]console.log("%c%s", "color:red", "message: "+this.message);   //今天是周五!!!},beforeUpdate(){console.group('beforeUpdate 更新前状态==========>>');console.log("%c%s", "color:red", "el     : "+this.$el);   //[object HTMLDivElement]console.log(this.$el);console.log(this.$el.innerHTML);    //<p>今天是周一!!!</p>console.log("%c%s", "color:red", "data   : "+this.$data); //[object Object]console.log("%c%s", "color:red", "message: "+this.message);   //今天周六了!!!},updated(){console.group('updated 更新完成状态==========>>');console.log("%c%s", "color:red", "el     : "+this.$el);   //[object HTMLDivElement]console.log(this.$el);console.log(this.$el.innerHTML);    //<p>今天周二了!!!</p>console.log("%c%s", "color:red", "data   : "+this.$data); //[object Object]console.log("%c%s", "color:red", "message: "+this.message);   //今天周六了!!!},beforeDestroy(){console.group('beforeDestroy 销毁前状态==========>>');console.log("%c%s", "color:red", "el     : "+this.$el);   //[object HTMLDivElement]console.log(this.$el);console.log("%c%s", "color:red", "data   : "+this.$data); //[object Object]console.log("%c%s", "color:red", "message: "+this.message);   //今天周六了!!!},destroyed(){console.group('destroyed 销毁完成状态==========>>');console.log("%c%s", "color:red", "el     : "+this.$el);   //[object HTMLDivElement]console.log(this.$el);console.log("%c%s", "color:red", "data   : "+this.$data); //[object Object]console.log("%c%s", "color:red", "message: "+this.message);   //今天周六了!!!},methods: {change(){this.message = "今天周二了!!!";console.group("==============点击事件执行的方法==============>>");console.log("%c%s", "color:red", "el     : "+this.$el);   //[object HTMLDivElement]console.log(this.$el);console.log("%c%s", "color:red", "data   : "+this.$data); //[object Object]console.log("%c%s", "color:red", "message: "+this.message);   //今天周六了!!!}}
})
</script>
</body>
</html>

函数执行顺序

1. 生命周期执行顺序

  • 页面初始化时:beforeCreate -> created -> beforeMount -> mounted
  • 页面发生修改时:beforeUpdate -> updated
  • 页面销毁时:beforeDestroy -> destroyed

2. 父子组件生命周期执行顺序

  • 页面初始化时:父beforeCreate -> 父created -> 父beforeMount ->子beforeCreate -> 子created -> 子beforeMount -> 子mounted-> 父mounted

从图中可以看到,子组件要先于父组件挂载完成。

  • 页面发生修改时:beforeUpdate -> updated 父、子组件间的更新互不影响,只更新自己。
  • 页面销毁时:父beforeDestroy -> 子beforeDestroy ->子destroyed->父 destroyed 销毁时也是子组件要先于父组件销毁

3. this.$nextTick在各生命周期的执行顺序

  • nextTick在各生命周期的执行顺序,nextTick是指在dom渲染完成后执行,结果如图。

  • 虽然每个周期使用$nextTick都可以获取到dom,但是还是建议在mounted中使用哈,因为beforeMount/mounted本来就是挂载dom滴

4. watch、computed、methods执行顺序

  • 页面初始化时: 会执行一次computed,watch初始化时不会执行,methods只有调用的时候才会执行。

  • 渲染完成后,触发methods: methods -> watch -> computed

总结

  • beforecreate:可以在这加个loading事件;
  • created :在这结束loading,还做一些初始化,实现函数自执行;
  • mounted : 在这发起后端请求,拿回数据,配合路由钩子做一些事情;
  • beforeDestory: 你确认删除vue实例了吗?
  • destoryed :当前实例已被销毁,解绑相关指令和事件监听器。

http://www.ppmy.cn/news/1512706.html

相关文章

Unity的动画系统

目录 Unity动画系统的最新更新和改进有哪些&#xff1f; 如何在Unity中高效地使用Animator组件进行复杂动画制作&#xff1f; Unity动画系统中的动画混合和分层功能是如何工作的&#xff1f; 动画混合&#xff08; blend tree&#xff09; 动画分层 在Unity中创建和管理动…

web后端(javaEE)开发——servlet

目录 一、web后端开发概述 二、web后端开发环境搭建 1.安装服务器软件 2.安装JDK 三、创建web后端项目 1.创建项目 2.修改设置 3.*在IDEA中集成Tomcat* 四、Servlet创建和应用 1.概述 2.Servlet程序创建与配置 3.分析Servlet程序结构 一、web后端开发概述 web开发&a…

YOLOv8轻量化改进之slimneck

目录 一、原理 二、代码 三、修改到YOLOv8中 四、yaml文件修改 一、原理 论文地址:2206.02424 (arxiv.org) 主要模块的网络结构 二、代码 slimneck的代码如下,slimneck主要由GSConv和VoVGSCSPC两部分组成。 class GSConv(nn.Module):# GSConv https://github.com/Alan…

SQL数据抽样:精准洞察的高效策略

标题&#xff1a; SQL数据抽样&#xff1a;精准洞察的高效策略 摘要&#xff1a; 在数据分析领域&#xff0c;数据抽样是一种有效减少数据集规模并提取代表性样本的方法。SQL作为数据查询和处理的标准语言&#xff0c;提供了多种数据抽样技术。本文将详细介绍SQL中的数据抽样方…

easycpp2-入土为安的第二十三天

考点&#xff1a;大小端&#xff0c;随机数&#xff0c;交换位置(>>) 查看导入表可以发现并没有导入srand, 所以rand的结果是固定的值, 可以自己编译一下查看rand()返回值是多少, 或者动态调试, 这里rotate_count的结果是rotate_count 7 func1_1是循环右移的逻辑, fun1_…

黑马头条vue2.0项目实战(十一)——功能优化(组件缓存、响应拦截器、路由跳转与权限管理)

1. 组件缓存 1.1 介绍 先来看一个问题&#xff1f; 从首页切换到我的&#xff0c;再从我的回到首页&#xff0c;我们发现首页重新渲染原来的状态没有了。 首先&#xff0c;这是正常的状态&#xff0c;并非问题&#xff0c;路由在切换的时候会销毁切出去的页面组件&#xff…

苍穹外卖项目DAY01

苍穹外卖项目Day01 1、软件开发整体介绍 1.1、软件开发流程 1.2、角色分工 项目经理&#xff1a;对整个项目负责&#xff0c;任务分配、把控进度产品经理&#xff1a;进行需求调研&#xff0c;输出需求调研文档、产品原型等UI设计师&#xff1a;根据产品原型输出界面效果图架…

第5天:常用的Python库和框架——Matplotlib:数据可视化

引言&#xff1a; Matplotlib是一种强大的Python数据可视化库&#xff0c;广泛应用于科学计算、数据分析和机器学习等领域。本文将介绍Matplotlib的基本概念和功能&#xff0c;以及提供几个实例代码来展示其在数据可视化方面的应用。 在数据分析和可视化领域&#xff0c;Matpl…