接着上一篇标准化props, inject, directives, 以及传入选项的extends, mixins属性的合并, 本篇来讲解mergeOptions函数的核心部分: 合并策略的定义。
我们看mergeOptions函数的最后一部分源码:
const options = {}
let key
for (key in parent) {
mergeField(key)
}
for (key in child) {
if (!hasOwn(parent, key)) {
mergeField(key)
}
}
function mergeField (key) {
const strat = strats[key] || defaultStrat
options[key] = strat(parent[key], child[key], vm, key)
}
复制代码
可以看到, 分别遍历parent和child对象, 对每一个key值都调用了mergeField()函数
mergeField()函数
是最终的合并策略函数。里面调用了strat()
函数, 而strat函数来自strats[key]
和defaultStrat
先看一下defaultStrat
/**
* Default strategy.
*/
const defaultStrat = function (parentVal: any, childVal: any): any {
return childVal === undefined
? parentVal
: childVal
}
复制代码
所以defaultStrat
的逻辑是,如果child上该属性值存在时,就取child上的该属性值,如果不存在,则取parent上的该属性值。优先取child上的选项配置。
再看strats[key]
strats是一个函数
const strats = config.optionMergeStrategies
export type Config = {
// user
optionMergeStrategies: { [key: string]: Function };
...
};
复制代码
可以在starts( config.optionMergeStrategies)上定义不同函数类型的key,然后定义这些key对应的合并策略。
config.optionMergeStrategies.el, config.optionMergeStrategies.propsData
if (process.env.NODE_ENV !== 'production') {
strats.el = strats.propsData = function (parent, child, vm, key) {
if (!vm) {
warn(
`option "${key}" can only be used during instance ` +
'creation with the `new` keyword.'
)
}
return defaultStrat(parent, child)
}
}
复制代码
从上述代码中可以看到, el以及propsData的合并策略是默认的合并策略, 如果子组件的选项存在, 就取子组件的,否则取父组件的。
####config.optionMergeStrategies.data、config.optionMergeStrategies.provide
strats.data = function (
parentVal: any,
childVal: any,
vm?: Component
): ?Function {
if (!vm) {
if (childVal && typeof childVal !== 'function') {
process.env.NODE_ENV !== 'production' && warn(
'The "data" option should be a function ' +
'that returns a per-instance value in component ' +
'definitions.',
vm
)
return parentVal
}
return mergeDataOrFn(parentVal, childVal)
}
return mergeDataOrFn(parentVal, childVal, vm)
}
strats.provide = mergeDataOrFn
复制代码
可以看到, data以及provide合并策略调用了mergeDataOrFn
函数; 以下分析data选项的合并策略, provide同理。
- 如果不是当前实例,即通过Vue.extend()创建的实例
- 如果childVal不是函数, 则返回parentVal作为当前data合并后的结果
- 否则调用
mergeDataOrFn(parentVal, childVal)
, 这个时候未传入vm实例
- 如果是当前实例,即通过new Vue()创建的实例
- 调用
mergeDataOrFn(parentVal, childVal, vm)
, 这个时候传入了vm实例
- 调用
找到mergeDataFn函数块
/**
* Data
*/
export function mergeDataOrFn (
parentVal: any,
childVal: any,
vm?: Component
): ?Function {
if (!vm) {
// in a Vue.extend merge, both should be functions
if (!childVal) {
return parentVal
}
if (!parentVal) {
return childVal
}
// when parentVal & childVal are both present,
// we need to return a function that returns the
// merged result of both functions... no need to
// check if parentVal is a function here because
// it has to be a function to pass previous merges.
return function mergedDataFn () {
return mergeData(
typeof childVal === 'function' ? childVal.call(this, this) : childVal,
typeof parentVal === 'function' ? parentVal.call(this, this) : parentVal
)
}
} else {
return function mergedInstanceDataFn () {
// instance merge
const instanceData = typeof childVal === 'function'
? childVal.call(vm, vm)
: childVal
const defaultData = typeof parentVal === 'function'
? parentVal.call(vm, vm)
: parentVal
if (instanceData) {
return mergeData(instanceData, defaultData)
} else {
return defaultData
}
}
}
}
复制代码
根据上面的分析很快我们就能找到:
- 如果没有传入vm当前实例, 也就是通过Vue.extend()/Vue.component创建的实例的时候
- 若没有childVal, 有parentVal, 则返回parentVal
- 若没有parentVal, 有childVal, 则返回childVal
- 若childVal,parentVal两者都有data选项, 则调用
mergeData()
函数合并data选项
- 如果传入了vm实例, 也就是通过new Vue()这种形式创建的实例
- 如果新建实例时传入了data选项,则调用
mergeData
函数合并实例和构造函数上的data选项 - 如果新建实例时没有传入data选项, 则返回构造函数上的data选项
- 如果新建实例时传入了data选项,则调用
不管是哪种方式创建的实例, 最终都调用了mergeData
函数进行合并data选项。我们看一下mergeData
函数
/**
* Helper that recursively merges two data objects together.
*/
function mergeData (to: Object, from: ?Object): Object {
if (!from) return to
let key, toVal, fromVal
const keys = hasSymbol
? Reflect.ownKeys(from)
: Object.keys(from)
for (let i = 0; i < keys.length; i++) {
key = keys[i]
// in case the object is already observed...
if (key === '__ob__') continue
toVal = to[key]
fromVal = from[key]
if (!hasOwn(to, key)) {
set(to, key, fromVal)
} else if (
toVal !== fromVal &&
isPlainObject(toVal) &&
isPlainObject(fromVal)
) {
mergeData(toVal, fromVal)
}
}
return to
}
复制代码
- 如果构造器(from)上没有data选项, 返回实例(to)上的data选项
- 如果构造器上也有data选项
- 先返回构造器from的data上所有的key值(keys)
- 遍历构造器data的keys
- 如果实例to的data选项上没有构造器data选项上的key值, 则调用set方法将该(key, fromVal)键值对挂到实例对象to的data选项里
- 否则, 如果to的data选项与构造器上的data选项有相同的key值, 并且该key对应的值是对象, 则递归调用mergeData函数
- 最后返回实例to上的data选项
####钩子函数合并的策略
export const LIFECYCLE_HOOKS = [
'beforeCreate',
'created',
'beforeMount',
'mounted',
'beforeUpdate',
'updated',
'beforeDestroy',
'destroyed',
'activated',
'deactivated',
'errorCaptured',
'serverPrefetch'
]
/**
* Hooks and props are merged as arrays.
*/
function mergeHook (
parentVal: ?Array<Function>,
childVal: ?Function | ?Array<Function>
): ?Array<Function> {
const res = childVal
? parentVal
? parentVal.concat(childVal)
: Array.isArray(childVal)
? childVal
: [childVal]
: parentVal
return res
? dedupeHooks(res)
: res
}
function dedupeHooks (hooks) {
const res = []
for (let i = 0; i < hooks.length; i++) {
if (res.indexOf(hooks[i]) === -1) {
res.push(hooks[i])
}
}
return res
}
LIFECYCLE_HOOKS.forEach(hook => {
strats[hook] = mergeHook
})
复制代码
我们看到,每一个钩子合并都是调用了mergeHook函数,mergeHook的逻辑
- 获取钩子数组res
- 如果child options上不存在这个钩子,parent存在, 就返回parent上的钩子
- 如果child, parent上都存在相同的钩子, 则返回concat之后的属性
- child options上存在, parent上不存在, 则判断child上的该属性是数组, 则直接返回child上该属性
- 如果res不存在, 返回res
- 否则返回dedupeHooks(res), 去重后的hooks
####Assets(components, directives, filters)的合并策略
/**
* Assets
*
* When a vm is present (instance creation), we need to do
* a three-way merge between constructor options, instance
* options and parent options.
*/
function mergeAssets (
parentVal: ?Object,
childVal: ?Object,
vm?: Component,
key: string
): Object {
const res = Object.create(parentVal || null)
if (childVal) {
process.env.NODE_ENV !== 'production' && assertObjectType(key, childVal, vm)
return extend(res, childVal)
} else {
return res
}
}
export const ASSET_TYPES = [
'component',
'directive',
'filter'
]
ASSET_TYPES.forEach(function (type) {
strats[type + 's'] = mergeAssets
})
复制代码
处理逻辑:如果child上存在这些选项,则与构造器parent上的选项合并, 否则直接返回构造器上的这些types选项
####props/methods/inject/computed的合并策略
/**
* Other object hashes.
*/
strats.props =
strats.methods =
strats.inject =
strats.computed = function (
parentVal: ?Object,
childVal: ?Object,
vm?: Component,
key: string
): ?Object {
if (childVal && process.env.NODE_ENV !== 'production') {
assertObjectType(key, childVal, vm)
}
if (!parentVal) return childVal
const ret = Object.create(null)
extend(ret, parentVal)
if (childVal) extend(ret, childVal)
return ret
}
复制代码
这个合并方法的逻辑也很简单:
- 如果构造器上没有该选项, 直接返回实例上的选项
- 如果构造器上有, 实例上也有, 返回合并后的结果
####watch的合并策略
/**
* Watchers.
*
* Watchers hashes should not overwrite one
* another, so we merge them as arrays.
*/
strats.watch = function (
parentVal: ?Object,
childVal: ?Object,
vm?: Component,
key: string
): ?Object {
// work around Firefox's Object.prototype.watch...
if (parentVal === nativeWatch) parentVal = undefined
if (childVal === nativeWatch) childVal = undefined
/* istanbul ignore if */
if (!childVal) return Object.create(parentVal || null)
if (process.env.NODE_ENV !== 'production') {
assertObjectType(key, childVal, vm)
}
if (!parentVal) return childVal
const ret = {}
extend(ret, parentVal)
for (const key in childVal) {
let parent = ret[key]
const child = childVal[key]
if (parent && !Array.isArray(parent)) {
parent = [parent]
}
ret[key] = parent
? parent.concat(child)
: Array.isArray(child) ? child : [child]
}
return ret
}
复制代码
逻辑处理同样也是三种:
- 如果实例上没有watch选项, 返回构造器parent上的watch选项或者null
- 否则如果实例上有, 构造器上没有, 就返回实例上的watch选项
- 否则两者都有的时候, 遍历实例上watch对象
- 如果parent上存在与child中watch的key
- 如果parent上该key对应的值不是数组, 则返回[parent]
- 合并parent和child的watch选项
- 如果parent上不存在与child中watch的key
返回数组形式的child
- 如果parent上存在与child中watch的key
终于熬完了合并选项的分析(三篇文章), 每天抽出睡觉的时间来啃这块, 非常不容易,促进自己成长, 希望也能给小可爱们带去一些启发, 如果有地方觉得说的不清楚的欢迎在下方评论区提问,讨论~
看到这里的小可爱们, 可顺手点个赞鼓励我继续创作, 创作不易,一起学习, 早日实现财富自由~