总结 Vue 知识体系之 12 个实用技巧
监听组件的生命周期
比如有父组件 Parent 和子组件 Child,如果父组件监听到子组件挂载 mounted 就做一些逻辑处理,常规的写法可能如下:
// Parent.vue
"doSomething"/>
// Child.vue
mounted() {
this.$emit("mounted");
}
此外,还有一种特别简单的方式,子组件不需要任何处理,只需要在父组件引用的时候通过@hook 来监听即可,代码如下:
"doSomething" /> <Child @hook:updated="doSomething" />
当然这里不仅仅是可以监听 mounted,其它的生命周期事件,例如:created,updated 等都可以。
watch 的初始立即执行
观察和响应 Vue 实例上的数据变动。类似于某些数据的监听回调 ,每当监听的数据变化时都会执行回调进行后续操作。
但是当 watch 一个变量的时候,初始化时并不会执行,如下面的例子,你需要在 created 的时候手动调用一次。
created() {
this.getList();
},
watch: {
keyWord: 'getList',
}
上面这样的做法可以使用,但很麻烦,我们可以添加 immediate 属性,这样初始化的时候就会自动触发(不用再写 created 去调用了),然后上面的代码就能简化为:
watch: {
keyWord: {
handler: 'getList',
immediate: true
}
}
watch 有三个参数
handler
:其值是一个回调函数。即监听到变化时应该执行的函数deep
:其值是 true 或 false;确认是否深入监听。immediate
:其值是 true 或 false,确认是否以当前的初始值执行 handler 的函数
路由参数变化组件不更新
同一path
的页面跳转时路由参数变化,但是组件没有对应的更新。
原因:主要是因为获取参数写在了created
或者mounted
路由钩子函数中,路由参数变化的时候,这个生命周期不会重新执行。
解决方案 1:watch
监听路由
watch: {
// 方法1 //监听路由是否变化
'$route' (to, from) {
if(to.query.id !== from.query.id){
this.id = to.query.id;
this.init();//重新加载数据
}
}
}
//方法 2 设置路径变化时的处理函数
watch: {
'$route': {
handler: 'init',
immediate: true
}
}
解决方案 2 :为了实现这样的效果可以给router-view
添加一个不同的key
,这样即使是公用组件,只要 url 变化了,就一定会重新创建这个组件。
"$route.fullpath"></router-view>
路由懒加载
Vue 项目中实现路由按需加载(路由懒加载)的 3 中方式:
// 1、Vue异步组件技术:
{
path: '/home',
name: 'Home',
component: resolve => reqire(['path路径'], resolve)
}
// 2、es6提案的import()
const Home = () => import('path路径')
// 3、webpack提供的require.ensure()
{
path: '/home',
name: 'Home',
component: r => require.ensure([],() => r(require('path路径')), 'demo')
}
require.context()
require.context(directory,useSubdirectories,regExp)
directory:说明需要检索的目录 useSubdirectories:是否检索子目录 regExp: 匹配文件的正则表达式,一般是文件名
场景:如页面需要导入多个组件,原始写法:
import titleCom from '@/components/home/titleCom'
import bannerCom from '@/components/home/bannerCom'
import cellCom from '@/components/home/cellCom'
components: {
titleCom, bannerCom, cellCom
}
这样就写了大量重复的代码,利用 require.context 可以写成
const path = require('path')
const files = require.context('@/components/home', false, /\.vue$/)
const modules = {}
files.keys().forEach((key) => {
const name = path.basename(key, '.vue')
modules[name] = files(key).default || files(key)
})
components: modules
递归组件
递归组件: 组件在它的模板内可以递归的调用自己,只要给组件设置 name 组件就可以了。 不过需要注意的是,必须给一个条件来限制数量,否则会抛出错误: max stack size exceeded 组件递归用来开发一些具体有未知层级关系的独立组件。比如:联级选择器和树形控件
<div v-for="(item,index) in treeArr"> {{index}} <br/>
<tree :item="item.arr" v-if="item.flag">tree>
div>
</template>