混入 (mixins) 是一种分发Vue组件中可复用功能的非常灵活的方式。混入对象可以包含任意组件选项。当组件使用混入对象时,所有混入对象的选项将被混入该组件本身的选项。
例子:(使用Vue.extend()
)
// 定义一个混入对象
var myMixin = {
created: function () {
this.hello()
},
methods: {
hello: function () {
console.log('hello from mixin!')
}
}
}
// 定义一个使用混入对象的组件
var Component = Vue.extend({
mixins: [myMixin]
})
var component = new Component() // => "hello from mixin!"
也可以在声明组件时使用extends
组件,无需使用Vue.extend,主要是为了扩展单文件组件:
var CompA = { ... }
// 在没有调用 `Vue.extend` 时候继承 CompA
var CompB = {
extends: CompA,
...
}
先定义一个基础组件CompBase
,作为混入对象:
This is a base mixin outside and I want to say {{text}}
Name: {{name}}
age: {{obj.age.value}}
然后在另外一个单文件组件中,声明一个组件
,这个组件由继承混入对象
而来。
demo20 Mixins
当组件和混入对象含有同名选项时,这些选项将以恰当的方式混合。
比如,数据对象在内部会进行浅合并(一层属性深度),在和组件的数据发生冲突时以组件数据优先。
在混入对象
中定义的数据对象会被
中定义的数据对象所替换,所以显示结果是:
This is a base mixin outside and I want to say bye
Name: jay
age: 10001
同名钩子函数将混合为一个数组,因此都将被调用。另外,混入对象的同名的钩子将在组件自身钩子之前调用,不同名的组件仍然按照生命周期的顺序执行。
值为对象的选项,例如methods
,components
和directives
,将被混合为同一个对象。两个对象键名冲突时,取组件对象的键值对。
中的mounted
钩子函数执行了methods
中的refresh
方法,
重新定义了mounted
钩子函数内容,重新定义了methods
中的refresh
方法,并且定义了created
钩子函数。
按照上面的规则,首先应按照生命周期的顺序执行函数,所以最先执行
的created
钩子函数,然后执行混入对象
中mounted
钩子,然后执行组件
的mounted
钩子。
执行执行混入对象
中mounted钩子时,对应的refresh
已经被
重新定义了,所以执行的是
中的refresh
函数。
因此,最终的执行顺序为:
'outer created'
'outer refresh'
'inner mounted'
'outer mounted'
也可以全局注册混入对象。注意使用! 一旦使用全局混入对象,将影响到所有之后创建的Vue实例
// 为自定义的选项 'myOption' 注入一个处理器。
Vue.mixin({
created: function () {
var myOption = this.$options.myOption
if (myOption) {
console.log(myOption)
}
}
})
new Vue({
myOption: 'hello!'
})
// => "hello!"
自定义选项将使用默认策略,即简单地覆盖已有值。如果想让自定义选项以自定义逻辑合并,可以向Vue.config.optionMergeStrategies
添加一个函数:
Vue.config.optionMergeStrategies.myOption = function (toVal, fromVal) {
// return mergedVal
}
对于大多数对象选项,可以使用methods
的合并策略:
var strategies = Vue.config.optionMergeStrategies
strategies.myOption = strategies.methods
更多高级的例子可以在Vuex的1.x混入策略里找到:
const merge = Vue.config.optionMergeStrategies.computed
Vue.config.optionMergeStrategies.vuex = function (toVal, fromVal) {
if (!toVal) return fromVal
if (!fromVal) return toVal
return {
getters: merge(toVal.getters, fromVal.getters),
state: merge(toVal.state, fromVal.state),
actions: merge(toVal.actions, fromVal.actions)
}
}