理解Vue.mixin,利用Vue.mixin正确的偷懒
关于 Vue.mixin 在 vue 官方文档中是这么解释的:
混入 (mixin) 提供了一种非常灵活的方式,来分发 Vue 组件中的可复用功能。一个混入对象可以包含任意组件选项。当组件使用混入对象时,所有混入对象的选项将被“混合”进入该组件本身的选项。
我们的理解:Vue.mixin 给我们提供了一种混入 Vue 实例的方法,创建了混入对象之后,我们自定义的方法或者变量可以很轻松的挂载在 Vue 实例上,给我们的偷懒带来方便;
Vue.mixin 为我们提供了两种混入方式:局部混入和全局混入;
本文还是以 demo 形式来进行学习讲解,如果有条件最好还是跟着 demo 敲一遍,这样印象才会深刻;
局部混入:
顾名思义就是部分混入,也就是只有引入了 mixin 的混入对象才可以使用,并且只有在引入了 mixin 混入对象的组件中才生效;
来,知道了概念,我们一起来看看代码:
首先自己搭建 Vue 的开发环境,然后我们在 src 目录中新建两个 vue 文件,分别是 page1.vue 和 page2.vue;
page1.vue
<template> <div>page1 的值是:</div> </template><script>
export default {
data () {
return {}
},
}
</script><style scoped>
</style>
page2.vue
<template> <div>page2 的值是:</div> </template><script>
export default {
data () {
return {}
}
}
</script><style scoped>
</style>
然后我们修改 App.vue
<template> <div id="app"> <button @click="method1">page1</button> <button @click="method2">page2</button><router-view></router-view>
</div>
</template><script>
export default {
name: 'App',
methods:{
method1(){
this.$router.push('/page1');
},
method2(){
this.$router.push('/page2');
}
}
}
</script><style>
#app {
font-family: 'Avenir', Helvetica, Arial, sans-serif;
-webkit-font-smoothing: antialiased;
-moz-osx-font-smoothing: grayscale;
text-align: center;
color: #2c3e50;
margin-top: 60px;
}
</style>
在 src 目录下创建 router.js 文件,配置路由实现跳转
import Vue from "vue"; import VueRouter from "vue-router"; Vue.use(VueRouter);import page1 from "./page1";
import page2 from "./page2";const routes=[
{path:"/page1",component:page1},
{path:"/page2",component:page2}
]const router=new VueRouter({
routes
})export default router
最后将路由引入 main.js 中:
// The Vue build version to load with the `import` command // (runtime-only or standalone) has been set in webpack.base.conf with an alias. import Vue from 'vue' import App from './App' import router from './router.js'Vue.config.productionTip = false
/ eslint-disable no-new /
new Vue({
el: '#app',
router,
components: {App},
template: '<App/>'
})
完成上述准备工作之后,我们可以看到现在的页面效果如下:
没有报错,我们开始正式进入学习 Vue.mixin:
首先我们在 src 目录下新建一个名为 mixin 的文件夹并在 mixin 文件中创建一个 mixin.js 文件:
//抛出混入对象,方便外部访问 export const mixin={data(){ return { number:1 } } }
可以看到我们在混入对象中创建了一个变量,是的,混入对象跟 Vue 实例的格式是一样的;
然后我们可以将 mixin.js 引入到我们的 page1.vue 和 page2.vue 中
page1.vue
<template> //这里读的值其实是 mixin 的值,因为这个时候 mixin 已经混入到 vue 实例中了 <div>page1 的值是:{{number}}</div> </template><script>
//引入 mixin.js
import {mixin} from "./mixin/mixin"
export default {
//这里注意:属性名为 mixins,值为数组类型
mixins:[mixin],
data () {
return {}
},
}
</script><style scoped>
</style>
page2.vue
<template> <div>page2 的值是:{{number}}</div> </template><script>
import {mixin} from "./mixin/mixin"
export default {
mixins:[mixin],
data () {
return {}
}
}
</script><style scoped>
</style>
这个时候我们的混入对象已经成功混入到 Vue 实例中,你们可以点击看看效果,是可以正常运行并且能读取到值的;
现在我们来修改 page1.vue 的代码:
<template> <div>page2 的值是:{{number}}</div> </template><script>
import {mixin} from "./mixin/mixin"
export default {
mixins:[mixin],
data () {
return {}
}
}
</script><style scoped>
</style>
page2 不变,再运行可以发现,我们的 page1.vue 中的值是执行了 mounted,所以产生了自增
由此,我们可以知道 mixin 混入对象的变量是不会共享的;也就是你 page1 发生了变化,并不会通知 mixin 进行实时刷新数据,发生的变化只会在 page1.vue 中生效,不影响其他组件;
现在我们修改 mixin.js 和 page1.vue 中的代码:
mixin.js
export const mixin={data(){ return { number:1 } }, created(){ console.log("mixin 混入对象")}}
page1.vue
<template> <div>page1 的值是:{{number}}</div> </template><script>
import {mixin} from "./mixin/mixin"
export default {
mixins:[mixin],
data () {
return {}
},
created(){
console.log("这里是 page1");
}
}
</script><style scoped>
</style>
这个时候我们再运行可以发现控制台输出是这个样子的:
是的,mixin 混入对象中声明了:如果是同名钩子函数将合并为一个数组,因此都被调用,但是混入对象的钩子将在自身实例钩子之前触发;
值为对象的选项,例如 methods,components 等如果变量名和 mixin 混入对象的变量名发生冲突,将会以组件优先并进行递归合并,相当于组件数据直接覆盖了 mixin 中的同名数据;
我们可以修改代码 mixin.js 和 page1.vue
mixin.js
export const mixin={data(){ return { number:1 } }, methods:{demo1(){ console.log("mixin 混入对象")}} }
page1.vue
<template> <div>page1 的值是:{{number}}</div> </template><script>
import {mixin} from "./mixin/mixin"
export default {
mixins:[mixin],
data () {
return {
number:10
}
},
mounted(){
this.demo1();
},
methods:{
demo1(){
console.log("这里是 page1");
}
}
}
</script><style scoped>
</style>
运行代码我们可以很清晰的看到都是执行我们组件内的值;
因为在 vue 中我们在实例中声明变量也是通过键值对的形式来声明的,其实也是一个对象;
全局混入:
全局混入我们只需要把 mixin.js 引入到 main.js 中,然后将 mixin 放入到 Vue.mixin() 方法中即可;
// The Vue build version to load with the `import` command // (runtime-only or standalone) has been set in webpack.base.conf with an alias. import Vue from 'vue' import App from './App' import router from './router.js' import mixin from "./mixin/mixin.js" Vue.config.productionTip = false Vue.mixin(mixin)/ eslint-disable no-new /
new Vue({
el: '#app',
router,
components: {App},
template: '<App/>'
})
是的,全局混入更为便捷,我们将不用在子组件声明,全局混入将会影响每一个组件的实例,使用的时候需要小心谨慎;这样全局混入之后,我们可以直接在组件中通过 this. 变量 / 方法来调用 mixin 混入对象的变量 / 方法;
很多同学可能看到这里会有一些疑问,这不就跟 Vuex 差不多嘛,其实不是的:
mixin 混入对象和 Vuex 的区别:
Vuex 是状态共享管理,所以 Vuex 中的所有变量和方法都是可以读取和更改并相互影响的;
mixin 可以定义公用的变量或方法,但是 mixin 中的数据是不共享的,也就是每个组件中的 mixin 实例都是不一样的,都是单独存在的个体,不存在相互影响的;
mixin 混入对象值为函数的同名函数选项将会进行递归合并为数组,两个函数都会执行,只不过先执行 mixin 中的同名函数;
mixin 混入对象值为对象的同名对象将会进行替换,都优先执行组件内的同名对象,也就是组件内的同名对象将 mixin 混入对象的同名对象进行覆盖;