‘壹’ Vue组件传值及页面缓存问题
关于父组件的传值类型和props更多的定义详见官网 : vue官网
(2)子组件向父组件传值
(3)通过 chlidren等方法调取用层级关系的组件内的数据和方法。
有很多时候根据业务需求要在同级组件或页面间传值,此处提供以下几种方法作为参考:
(1)通过router-link进行跳转
(2) this.$router.push()
此方法同样是有path+query和name+params两种方式:
总结:使用query,传输的值会在url后面以参数的形式显示出来,可以刷新页面,数据不变,但会是页面路由过长;而params只要一刷新传递的参数就没了。
(3)LocalStorage缓存传值
注意:简单的小项目可以这么做,如果项目很大,建议直接用vuex。
(4)通过Vuex进行传值
(5)发布订阅模式(也叫eventBus或事件总线)
在Vue的原型上定义一个变量eventBus,所有所有Vue的实例或组件都将共享这个eventBus,可以用eventBus来发布自定义事件,然后在组件中用eventBus订阅自定义事件。就可以实现传值。
详细讲解可看 链接
(6)Vue.observable
index.vue组件中触发:
Vue中如何在切换组件过程中,将状态保存到内存中,防止DOM重新渲染,通俗的讲就是实现如何在一个页面输入部分数据后到了另一个页面再返回该页面,数据还在。
需求分析:Page1中录入信息,页面跳转带Page2,然后再返回Page1,之前Page1录入的信息还存在。
现在更改需求为:
首页是A页面
A页面跳转到B,B页面不需要缓存
B页面跳转到C,C页面不需要被缓存
C页面返回到B,B页面需要缓存
B页面返回到A,
A再次跳转到B
(1)此时思路是在每个路由的beforeRouteLeave(to, from, next)钩子中设置to.meta.keepAlive
beforeRouteLeave讲解
PageA页面:
PageB页面:
(2)用eventBus解决此问题
需要注意的一点是发布订阅第一次会无效,因为订阅的组件还没创建。解决方法就是首次进入pageB页面时接收pageA页面params里传递的参数。
‘贰’ vue中动态路由组件缓存及生命周期函数
<keep-alive>是Vue的内置组件,能在组件切换过程中将状态保留在内存中,防止重复渲染DOM。<keep-alive> 包裹动态组件时,会缓存不活动的组件实例,而不是销毁它们。和 <transition> 相似,<keep-alive> 是一个抽象组件:它自身不会渲染一个 DOM 元素,也不会出现在父组件链中。prop :
有两个生命周期函数:
activated :激活
deactivated :失活
利用keep-alive实现滚动条保存:
思路 : 在路由页面离开 beforeRouterEnter 中保存当前滚动条距离顶部的位置,再用 activated 钩子函数 当他再次被激活时,让他的滚动条等于离开时保存的那个值。
beforeCreate :组件实例刚被创建,组件属性计算之前,如data属性等
created :组件实例创建完成,属性已绑定,但DOM还未生成,$el属性还不存在
beforeMount :模板编译 / 挂载之前
mounted :模板编译 / 挂载之后
beforeUpdate :组件更新之前
updated :组件更新之后
activated :组件被激活时调用
deactivated :组件被移除时调用
beforeDestory :组件销毁前调用
destoryed :组件销毁后调用
(1) 创建阶段
创建的标志点是New vue(),beforeCreate和created都发生在创建动作之后,但区别在于beforeCreate触发的时候数据还没初始化和绑定,而created的时候就生成好了。
(2) 挂载阶段
beforeMount和mounted两者主要区别在于模板是否编译和挂载了。
(3) 更新阶段
beforeUpdate 和 updated 就是当数据发生变化的时候出发的。
(4) 销毁阶段
beforeDestory 和 destoryed 的区别就是el的值data的数据依然在的。这是因为$destroy只是销毁一个实例,清理它与其它实例的连接,解绑它的全部指令及事件监听器,并不会清除data的数据或者清除dom。
‘叁’ vue 路由缓存
在写一个移动端项目时遇到了一个问题,每一个tab栏所对应的页面被封装成一个组件进行
复用,结果会导致他们共用一个滚动条。解决办法是给这个子组件加固定的高度(例如
height:100vh),同时加一个属性overflow-y:auto,这样就可以解决这个问题。之后发现每当进
入主页中任意一篇文章再退出时,页面重新加载渲染了,之前的文章消失了,所有在这里想到
了路由缓存。
先说一下keep-alive的属性和用法。
Props :
用法 :
<keep-alive> 包裹动态组件时,会缓存不活动的组件实例,而不是销毁它们。
<keep-alive> 是一个抽象组件:它自身不会渲染一个 DOM 元素,也不会出现在组件的父组件链中。
当组件在 <keep-alive> 内被切换,它的 activated 和 deactivated 这两个生命周期钩子函数将会被对应执行。
(只有被<keep-alive>包裹的组件才具有这两个钩子函数)
其实不太建议使用include的,因为如果多个组件中都包括include中匹配到的组件,想单独
的让其中某个组件缓存是不太方便的。所以可以在分发路由时,在需要缓存的路由中把额外标
记的属性写在meta中,如图1。
使用的时候可以在根组件中通过$route.meta.keepAlive取到该值,然后进行v-if判断即可,如图
2,这样使用起来相对灵活一点。
使用keep-alive缓存路由后,已经解决了主要问题,但是有新的问题出现。缓存过的组件重
新被激活时不会记录滚动条的状态,默认置顶,所以应该监听这个被复用的组件的onscroll事
件,动态记录滚动条的位置状态,然后重新赋值给scrollTop(如果还想优化最好给滚动事件来
一个防抖处理)。那么问题来了,应该在何时重新赋值呢?上面用法中提到了当组件在<keep-
alive> 内被切换时会触发activated这个钩子函数,所以应在这里赋值。
‘肆’ vue的dom更新缓冲确保无论你进行了多少次声明更改每个组件都只需要更新1次吗
Vue 的 DOM 更新缓冲机制确保了当你进行多次声明更改时,每个组件最多只会更新一次局棚。它通过使亮让用异步队列来缓冲更新操作,等到所有的更改都被记录下来之后再一次性更新 DOM。这桐键则样做的好处是可以减少不必要的 DOM 操作,提高性能。
Vue 是通过一种叫做脏检查(dirty checking)的机制来实现这一点的。它会在每次更新时扫描所有的组件并检查它们的状态是否发生了变化,如果发现了变化就会更新对应的 DOM。这个过程被称为脏检查。
但是,如果每次更新都要扫描整个应用的所有组件,时间复杂度将会是 O(n) 级别的,这显然是不可取的。因此 Vue 会对组件进行分组,只对变化了的组件进行脏检查。
总结来说, Vue 的 DOM 更新缓冲机制确保了不管你进行了多少次声明更改,每个组件都最多只会更新一次。这样做的好处是可以减少不必要的 DOM 操作,提高性能。