解析構(gòu)造函數(shù)的options

前言

這章其實(shí)只講src/core/instance/init.js里的resolveConstructorOptions函數(shù)者祖。
這個(gè)函數(shù)水有點(diǎn)深怎茫,網(wǎng)上的很多文章都沒(méi)說(shuō)全劳曹,所以單獨(dú)拎出來(lái)

正文

首先來(lái)個(gè)栗子

const fn1 = function fn1() {
    console.log('fn1')
}
const fn2 = function fn2() {
    console.log('fn2')
}
const fn3 = function fn3() {
    console.log('fn3')
}
const fn4 = function fn4() {
    console.log('fn4')
}
const Parent = Vue.extend({
    template: '<p>1</p>',
    created: [fn1]
})
const Child = Parent.extend({
    name: 'Child',
    template: '<p>2</p>',
    created: [fn2]
})
Vue.mixin({
    template: '<p>3</p>',
    created: [fn3]
})
Child.mixin({
    template: '<p>4</p>',
    created: [fn4]
})
new Child({}).$mount('#app')

當(dāng)我們計(jì)算Childoptions的時(shí)候我們不能簡(jiǎn)單的取Child.options温艇,因?yàn)楹竺嫫涓父割?code>Vue混入了options逻恐,其本身也混入了options像吻。這時(shí)候我們?nèi)〉?code>Child.options會(huì)漏了Vue.mixin混入的options峻黍。
這也是很好理解,因?yàn)?code>Child.options在Child = Parent.extend()之后除了Child.mixin()就沒(méi)改過(guò)拨匆,但是Vue.mixin()導(dǎo)致Vue.options改變了姆涩,所以本該繼承下來(lái)的沒(méi)繼承

resolveConstructorOptions就是為了解決這個(gè)問(wèn)題,因?yàn)榫统鲈诟割惖淖兓衙浚运乃枷刖褪前雅袛喔割惖?code>options是否變化骨饿,變化了的話就把繼承下來(lái)的.extend、.mixin擴(kuò)展的區(qū)分開(kāi)來(lái)台腥,后者算新增的宏赘,前者是繼承的。倆者合并就是新的options

resolveConstructorOptions

就像標(biāo)題所述黎侈,這個(gè)是解析傳入的構(gòu)造函數(shù)的options察署,然后返回新的options(內(nèi)部會(huì)修正一些變量)
首先我們假設(shè)下面的Ctor是在Child的時(shí)候的情況對(duì)于上文的栗子而言

export function resolveConstructorOptions(Ctor: Class < Component > ) {
    let options = Ctor.options
    if (Ctor.super) {
        const superOptions = resolveConstructorOptions(Ctor.super)
        const cachedSuperOptions = Ctor.superOptions
        if (superOptions !== cachedSuperOptions) {
            Ctor.superOptions = superOptions
            const modifiedOptions = resolveModifiedOptions(Ctor)
            if (modifiedOptions) {
                extend(Ctor.extendOptions, modifiedOptions)
            }
            options = Ctor.options = mergeOptions(superOptions, Ctor.extendOptions)
            if (options.name) {
                options.components[options.name] = Ctor
            }
        }
    }
    return options
}

首先根據(jù)是否有super屬性來(lái)判斷是否是子類
若是是子類的話看看分支之內(nèi)

/**
{
    components: {},
    directives: {},
    filters: {},
    _base: function () {},
    template: "<p>1</p>",
    created: [fn3, fn1]
}
*/
const superOptions = resolveConstructorOptions(Ctor.super)
/**
{
    components: {},
    directives: {},
    filters: {},
    _base: function () {},
    template: "<p>1</p>",
    created: [fn1]
}
*/
const cachedSuperOptions = Ctor.superOptions

首先獲取父類(Parent)正確options,然后獲取執(zhí)行Child = Parent.extend()時(shí)緩存的父類(Parent)options峻汉。因?yàn)榍罢呤钱?dāng)前父類(Parent)真實(shí)的options贴汪,所以倆者可以通過(guò)比較來(lái)判斷之后是否改變過(guò)。
很明顯改過(guò)了(Vue.mixin({...})導(dǎo)致Parent繼承的值變了休吠,也就導(dǎo)致其真實(shí)的options變了)
若是有改動(dòng)的話就先修正

Ctor.superOptions = superOptions

然后到重要的點(diǎn)

const modifiedOptions = resolveModifiedOptions(Ctor)

就是它扳埂,解析新增的那部分options

resolveModifiedOptions

這個(gè)就是上面所說(shuō)的獲取通過(guò).extend、.mixin新增的那部分options

function resolveModifiedOptions(Ctor: Class < Component > ): ? Object {
    let modified
    /**
    {
        components: {},
        directives: {},
        filters: {},
        _base: function () {},
        template: "<p4</p>",
        created: [fn1, fn2, fn4]
      }
    */
    const latest = Ctor.options
    // { template: '<p>2</p>', created: [fn2] }
    const extended = Ctor.extendOptions
    /**
    {
        components: {},
        directives: {},
        filters: {},
        _base: function () {},
        template: "<p>2</p>",
        created: [fn1, fn2]
      }
    */
    const sealed = Ctor.sealedOptions
    for (const key in latest) {
        if (latest[key] !== sealed[key]) {
            if (!modified) modified = {}
            modified[key] = dedupe(latest[key], extended[key], sealed[key])
        }
    }
    return modified
}

首先獲取三個(gè)屬性瘤礁,我們單看created

  • latest 最新的options阳懂。首先fn1Parent繼承而來(lái),fn2是生成自身時(shí)傳入的參數(shù),fn4Child.mixin混入的,至于fn3Vue上几缭,相對(duì)Child是祖父糊肤,不過(guò)因?yàn)槭?code>Parent生成之后才混入,所以就沒(méi)繼承到,所以需要修正
  • extended 執(zhí)行.extend傳入的的options參數(shù)。這個(gè)很簡(jiǎn)單,就是生成Child時(shí).extend傳入的參數(shù)堕澄,也就是只有fn2
  • sealed 執(zhí)行.extend時(shí)options的數(shù)據(jù)副本(若是之后有變動(dòng),那么這個(gè)值將和.options的值不一樣)霉咨。這個(gè)也簡(jiǎn)單蛙紫,就是生成Child時(shí)Child.options數(shù)據(jù)副本(也就是之后只要沒(méi)修正都不會(huì)變動(dòng),所以叫sealed)途戒。fn1Parent繼承而來(lái)坑傅,fn2是生成自身時(shí)傳入的參數(shù)

然后遍歷latest,要是值有了變化

if (latest[key] !== sealed[key]) {
    if (!modified) modified = {}
    modified[key] = dedupe(latest[key], extended[key], sealed[key])
}

那么就得判斷這個(gè)值是否有重復(fù)(數(shù)組情況喷斋,比如生命周期鉤子)唁毒,這個(gè)所謂的重呢就是是否和父類繼承過(guò)來(lái)的那部分重復(fù)了蒜茴。很明顯,這里的fn1就重了浆西,因?yàn)槭?code>Parent繼承來(lái)的

dedupe

去重函數(shù)

function dedupe(latest, extended, sealed) {
    if (Array.isArray(latest)) {
       ...
    } else {
        return latest
    }
}

首先判斷傳入的當(dāng)前值是不是數(shù)組粉私,如果不是,那么就直接返回最新值近零,否則的話

const res = []
// [fn1, fn2]
sealed = Array.isArray(sealed) ? sealed : [sealed]
// [fn2]
extended = Array.isArray(extended) ? extended : [extended]
for (let i = 0; i < latest.length; i++) {
    if (extended.indexOf(latest[i]) >= 0 || sealed.indexOf(latest[i]) < 0) {
        res.push(latest[i])
    }
}
// [fn2, fn4]
return res

我們可見(jiàn)首先規(guī)范化倆參數(shù)為數(shù)組诺核,然后遍歷最新值,這里這個(gè)if語(yǔ)句有點(diǎn)難理解
其實(shí)他走的倆策略

  • 若是通過(guò).extend傳入的那么就不是繼承來(lái)的
  • 若不在sealed久信,那么必然是.extend之后改動(dòng)的窖杀,也是新增的
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市裙士,隨后出現(xiàn)的幾起案子入客,更是在濱河造成了極大的恐慌,老刑警劉巖腿椎,帶你破解...
    沈念sama閱讀 206,482評(píng)論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件痊项,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡酥诽,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,377評(píng)論 2 382
  • 文/潘曉璐 我一進(jìn)店門(mén)皱埠,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)肮帐,“玉大人,你說(shuō)我怎么就攤上這事边器⊙凳啵” “怎么了?”我有些...
    開(kāi)封第一講書(shū)人閱讀 152,762評(píng)論 0 342
  • 文/不壞的土叔 我叫張陵忘巧,是天一觀的道長(zhǎng)恒界。 經(jīng)常有香客問(wèn)我,道長(zhǎng)砚嘴,這世上最難降的妖魔是什么十酣? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 55,273評(píng)論 1 279
  • 正文 為了忘掉前任,我火速辦了婚禮际长,結(jié)果婚禮上耸采,老公的妹妹穿的比我還像新娘。我一直安慰自己工育,他們只是感情好虾宇,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,289評(píng)論 5 373
  • 文/花漫 我一把揭開(kāi)白布。 她就那樣靜靜地躺著如绸,像睡著了一般嘱朽。 火紅的嫁衣襯著肌膚如雪旭贬。 梳的紋絲不亂的頭發(fā)上,一...
    開(kāi)封第一講書(shū)人閱讀 49,046評(píng)論 1 285
  • 那天搪泳,我揣著相機(jī)與錄音稀轨,去河邊找鬼。 笑死森书,一個(gè)胖子當(dāng)著我的面吹牛靶端,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播凛膏,決...
    沈念sama閱讀 38,351評(píng)論 3 400
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼杨名,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了猖毫?” 一聲冷哼從身側(cè)響起台谍,我...
    開(kāi)封第一講書(shū)人閱讀 36,988評(píng)論 0 259
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎吁断,沒(méi)想到半個(gè)月后趁蕊,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 43,476評(píng)論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡仔役,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 35,948評(píng)論 2 324
  • 正文 我和宋清朗相戀三年掷伙,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片又兵。...
    茶點(diǎn)故事閱讀 38,064評(píng)論 1 333
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡任柜,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出沛厨,到底是詐尸還是另有隱情宙地,我是刑警寧澤,帶...
    沈念sama閱讀 33,712評(píng)論 4 323
  • 正文 年R本政府宣布逆皮,位于F島的核電站宅粥,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏电谣。R本人自食惡果不足惜秽梅,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,261評(píng)論 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望剿牺。 院中可真熱鬧风纠,春花似錦、人聲如沸牢贸。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 30,264評(píng)論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至臭增,卻和暖如春懂酱,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背誊抛。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 31,486評(píng)論 1 262
  • 我被黑心中介騙來(lái)泰國(guó)打工列牺, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人拗窃。 一個(gè)月前我還...
    沈念sama閱讀 45,511評(píng)論 2 354
  • 正文 我出身青樓瞎领,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親随夸。 傳聞我的和親對(duì)象是個(gè)殘疾皇子九默,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,802評(píng)論 2 345

推薦閱讀更多精彩內(nèi)容

  • Vue 實(shí)例 屬性和方法 每個(gè) Vue 實(shí)例都會(huì)代理其 data 對(duì)象里所有的屬性:var data = { a:...
    云之外閱讀 2,202評(píng)論 0 6
  • 前言 使用Vue在日常開(kāi)發(fā)中會(huì)頻繁接觸和使用生命周期,在官方文檔中是這么解釋生命周期的: 每個(gè) Vue 實(shí)例在被創(chuàng)...
    心_c2a2閱讀 2,233評(píng)論 1 8
  • 回憶 首先,render函數(shù)中手寫(xiě)h=>h(app)宾毒,new Vue()實(shí)例初始化init()和原來(lái)一樣驼修。$mou...
    LoveBugs_King閱讀 2,270評(píng)論 1 2
  • new Vue 的過(guò)程通常有 2 種場(chǎng)景,一種是我們的代碼主動(dòng)調(diào)用 new Vue(options) 的方式實(shí)例化...
    小螃蟹_5f4c閱讀 234評(píng)論 0 0
  • 我總覺(jué)得诈铛,人活在世上是很苦的乙各。就像是小心翼翼地走在一條很長(zhǎng)很長(zhǎng)的潮濕的河邊,抬起頭幢竹,看見(jiàn)的是幽暗的層云耳峦,低下頭,看...
    秋豆子閱讀 223評(píng)論 0 0