接着上一篇標準化props, inject, directives, 以及傳入選項的extends, mixins屬性的合併, 本篇來說解mergeOptions函數的核心部分: 合併策略的定義。數組
咱們看mergeOptions函數的最後一部分源碼:markdown
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()函數
ide
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上的選項配置。fetch
再看strats[key]
ui
strats是一個函數this
const strats = config.optionMergeStrategies
export type Config = {
// user
optionMergeStrategies: { [key: string]: Function };
...
};
複製代碼
能夠在starts( config.optionMergeStrategies)上定義不一樣函數類型的key,而後定義這些key對應的合併策略。spa
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的合併策略是默認的合併策略, 若是子組件的選項存在, 就取子組件的,不然取父組件的。prototype
####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同理。
mergeDataOrFn(parentVal, childVal)
, 這個時候未傳入vm實例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
}
}
}
}
複製代碼
根據上面的分析很快咱們就能找到:
mergeData()
函數合併data選項mergeData
函數合併實例和構造函數上的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
}
複製代碼
####鉤子函數合併的策略
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的邏輯
####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
}
複製代碼
邏輯處理一樣也是三種:
終於熬完了合併選項的分析(三篇文章), 天天抽出睡覺的時間來啃這塊, 很是不容易,促進本身成長, 但願也能給小可愛們帶去一些啓發, 若是有地方以爲說的不清楚的歡迎在下方評論區提問,討論~
看到這裏的小可愛們, 可順手點個贊鼓勵我繼續創做, 創做不易,一塊兒學習, 早日實現財富自由~