vue 作为目前前端三大框架之一,对于前端开发者可以说是必备技能。掌握这些实用小技巧,可以让你事半功倍。
路由懒加载可以让我们的包不需要一次把所有的页面的加载进来,只加载当前页面的路由组件就行。
举个栗子,如果这样写,加载的时候会全部都加载进来。
const router = new VueRouter({
routes:[
{
path: '/',
name: 'Home',
component: Home
},
{
path: '/about',
name: 'About',
component: About
}
]
})
所以,应该避免上面的写法,尽量使用懒加载。
路由的懒加载可以分为以下三种写法。
// 1、Vue异步组件
VueRouter({
routes:[
{
path: '/about',
name: 'About',
component: resolve => reqire(['path路径'], resolve)
}
]
})
// 2、es6的import
VueRouter({
routes:[
{
path: '/about',
name: 'About',
component: () => import('path路径')
}
]
})
// 3、webpack提供的require.ensure()
VueRouter({
routes:[
{
path: '/about',
name: 'About',
component: r => require.ensure([],() => r(require('path路径')), 'demo')
}
]
})
异步组件可以让我们在需要一些组件时才将它加载进来,而不是一初始化就加载进来,这跟路由懒加载是一个概念。
以前是这么引入组件的
import BureauDetail from './components/ChildFirst'
import addBureau from './components/ChildSecond'
//在vue的comoinents中
components: {
ChildFirst,
ChildSecond
}
如果不是一开始就要加载的组件,我们可以使用组件懒加载
//在vue的comoinents中
components: {
BureauDetail: () => import('./components/ChildFirst'),
addBureau: () => import('./components/ChildSecond')
},
异步组件还有一种比较完善的写法
export default {
components:{
ChildFirst:()=>({
component:import(/* webpackChunkName: "ChildFirst" */ './Async'),
delay:200, // 延迟几毫秒,默认200
timeout:3000, // 加载几毫米之后就超时,触发error组件
loading:LoadingComponent, // 组件未加载回来前显示
error:ErrorComponent // 组件超时时显示
})
}
}
常常用来在组件内引入多个组件, require.context(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
在main.js中引入大量公共组件,利用 require.context 可以写成
import Vue from 'vue'
// 自定义组件
const requireComponents = require.context('../views/components', true, /\.vue/)
// 打印结果
// 遍历出每个组件的路径
requireComponents.keys().forEach(fileName => {
// 组件实例
const reqCom = requireComponents(fileName)
// 截取路径作为组件名
const reqComName =reqCom.name|| fileName.replace(/\.\/(.*)\.vue/,'$1')
// 组件挂载
Vue.component(reqComName, reqCom.default || reqCom)
})
比如有父组件 Parent 和子组件 Child,如果父组件监听到子组件挂载 mounted 就做一些逻辑处理,常规的写法可能如下:
// Parent.vue
<Child @mounted="doSomething"/>
// Child.vue
mounted() {
this.$emit("mounted");
}
此外,还有一种特别简单的方式,子组件不需要任何处理,只需要在父组件引用的时候通过@hook 来监听即可,@hook也可以监听其它的生命周期事件,代码如下:
<Child @hook:mounted="doSomething" />
<Child @hook:updated="doSomething" />
在computed属性中通过this.xxx去拿data里面的数据,和methods里面的方法吧,或许还会通过this.$store去拿vuex的state,和commit等,甚至,还会通过this.$route去获取路由里面的数据吧。
其实,我们可以避免这些丑陋的this,它甚至会给我们带来看不见的性能问题。
实现上,我们通过this能访问到的数据,在computed的第一个参数上都能结构出来。
export default {
watch: {
haha({$attrs,$route,$store,$listeners,$ref}){
// 还能结构很多属性,可自行打印看看
return
}
}
}
watch每当监听的数据变化时都会执行回调进行后续操作。
但是当 watch 一个变量的时候,在页面或者组件初始化时并不会执行,如下面的例子,你需要在 created 的时候手动调用一次。
created() {
this.getList();
},
watch: {
keyWord: 'getList',
}
上面这样的做法可以使用,但很麻烦,我们可以添加 immediate 属性,这样初始化的时候就会自动触发
watch 有三个参数
watch: {
keyWord: {
handler(val) {},
immediate: true
}
}
递归组件: 组件在它的模板内可以递归的调用自己,只要给组件设置 name 组件就可以了。
注意:必须给一个条件来限制数量,否则会抛出错误: max stack size exceeded
<template>
<div v-for="(item,index) in treeArr"> {{index}} <br/>
<tree :item="item.arr" v-if="item.flag"></tree>
</div>
</template>
<script>
export default {
// 必须定义name,组件内部才能递归调用
name: 'tree',
data(){
return {}
},
// 接收外部传入的值
props: {
item: {
type:Array,
default: ()=>[]
}
}
}
</script>
看Element ui源码时,封装的tree控件、级联选择其,也是这么使用的。
vue 2.0版本会通过 object.defineProperty 对数据进行劫持,遇到数组和对象必须循环遍历所有的域值才能劫持每一个属性。
vue 3.0版本会通过Proxy构造函数来进行数据劫持,来实现视图响应数据的变化
然而有些时候我们的组件就是纯粹的数据展示,不会有任何改变,我们就不需要 vue 来劫持我们的数据,在大量数据展示的情况下,这能够很明显的减少组件初始化的时间。
所以,我们可以通过 object.freeze 方法来冻结一个对象,这个对象一旦被冻结,vue就不会对数据进行劫持了。
Object.freeze() 可以冻结一个对象,冻结之后不能向这个对象添加新的属性,不能修改其已有属性的值,不能删除已有属性,以及不能修改该对象已有属性的可枚举性、可配置性、可写性。该方法返回被冻结的对象。
<p v-for="item in list">{{ item.value }}</p>
export default {
data: {
// vue不会对list里的object做getter、setter绑定
list: Object.freeze([
{ value: 1 },
{ value: 2 }
])
},
created () {
// 界面不会有响应
this.list[0].value = 100;
// 下面两种做法,界面都会响应
this.list = [
{ value: 100 },
{ value: 200 }
];
this.list = Object.freeze([
{ value: 100 },
{ value: 200 }
]);
}
}
当vue的data里边声明或者已经赋值过的对象或者数组(数组里边的值是对象)时,向对象中添加新的属性,如果更新此属性的值,是不会更新视图的。
调用方法: this.$set( target , key , value)
export default {
name: 'App',
data () {
return {
items: [
{ message: "one", id: "1" },
{ message: "two", id: "2" },
{ message: "three", id: "3" }
]
}
},
mounted () {
//此时对象的值更改了,但是视图没有更新
this.items[0] = { message:'first',id:'4'}
//$set 可以触发更新视图
this.$set(this.items,0,art)
},
methods: {
handClick(){
let change = this.items[0]
change.message="shen"
//$set 可以触发更新视图
this.$set(this.items,0,change)
}
}
}
</script>
.sync提供了一种与父组件沟通的思路!你如果只是单纯的在子组件当中修改父组件的某个数据时,建议使用sync,简单,快捷,不需要在传一个自定义方法来接收了
vue中我们经常会用v-bind(缩写为:)给子组件传入参数。或者我们会给子组件传入一个函数,子组件通过调用传入的函数来改变父组件的状态。举个例子
//父组件 给子组件传入一个函数
<MyFooter :age="age" @setAge="(res)=> age = res">
</MyFooter>
//子组件 通过调用这个函数来实现修改父组件的状态。
mounted () {
console.log(this.$emit('setAge',1234567));
}
现在只需要使用.sync就可以轻松更新赋组件的值
//父组件 将age传给子组件并使用.sync修饰符。
<MyFooter :age.sync="age">
</MyFooter>
//子组件 触发事件
mounted () {
console.log(this.$emit('update:age',1234567));
}
混入 (mixin) 提供了一种非常灵活的方式,来分发 Vue 组件中的可复用功能。一个混入对象可以包含任意组件选项。当组件使用混入对象时,所有混入对象的选项将被“混合”进入该组件本身的选项。
通俗的说,mixin可以定义公用的data,created,methods,computed,watch等。然后混入到你当前的vue文件中。
minxin还有一个强大之处就是合并选项,相同的变量/方法名会合并在一起,如果有相同名字,当前文件的变量或者方法会覆盖mixin文件的名字或者方法。
1、定义一个 mixin.js
export default mixin {
data() {
return {
name: 'mixin'
}
},
created() {
console.log('mixin...', this.name);
},
mounted() {},
methods: { //日期转换
formatDate (dateTime, fmt = 'YYYY年MM月DD日 HH:mm:ss') {
if (!dateTime) {
return ''
}
moment.locale('zh-CN')
dateTime = moment(dateTime).format(fmt)
return dateTime
}
}
}
2、在vue文件中使用mixin
import '@/mixin'; // 引入mixin文件
export default {
mixins: [mixin], //用法
data() {
return {
userName: "adimin",
time: this.formatDate(new Date()) //这个vue文件的数据源data里面的time就是引用混入进来的方法
}
}
}
常用的父子组件通信方式都是父组件绑定要传递给子组件的数据,子组件通过props属性接收,一旦组件层级变多时,采用这种方式一级一级传递值非常麻烦,而且代码可读性不高,不便后期维护。
vue提供了provide和inject帮助我们解决多层次嵌套嵌套通信问题。在provide中指定要传递给子孙组件的数据,子孙组件通过inject注入祖父组件传递过来的数据。
其实,provide 和 inject 主要为高阶插件/组件库提供用例。
读Element UI源码时,发现很多高阶组件使用了provide和inject,减少了很多父组件向子组件以及孙子组件一层层通信的工作量。
elementUI组件库中,在el-form组件中将组件实例暴露给子孙组件
在el-form-item组件中注入el-form组件实例,然后就可以使用el-form组件实例的方法、变量等等
1、在父组件中provide提供变量
<template>
<div>
<p>{{ title }}</p>
<son></son>
</div>
</template>
<script>
import Son from "./son"
export default {
name: 'Father',
components: { Son },
// provide选项提供变量
provide: {
message: 'provided by father'
},
data () {
return {
title: '父组件'
}
},
methods: { ... }
}
</script>
2、在子孙组件中,都可以使用inject来注入
<template>
<div>
<p>message:{{ message }}</p>
</div>
</template>
<script>
export default {
name: "GrandSon",
inject: [ "message" ],
data () {
return {
title: '孙组件'
}
},
methods: { ... }
};
</script>
Copyright© 2013-2020
All Rights Reserved 京ICP备2023019179号-8