VUE源码学习-全局api初始化

源码阅读路径src/core/global-api/index.js

此文件主要做了以下几件事:
  1. 定义Vue.config的属性,如下图所示(Vue.config.png)
  2. Vue.util上定义一些方法,(慎用)
  3. 定义全局方法 Vue.setVue.deleteVue.nextTickVue.observable
  4. 初始化Vue.options[components | directives | filters]为空
  5. 增加全局组件(keep-alive、Transition)
  6. 定义全局方法Vue.useVue.mixinVue. extendVue. componentVue. filterVue. directive
    Vue.config.png

入口文件

import config from '../config'
import { initUse } from './use'
import { initMixin } from './mixin'
import { initExtend } from './extend'
import { initAssetRegisters } from './assets'
import { set, del } from '../observer/index'
import { ASSET_TYPES } from 'shared/constants'
import builtInComponents from '../components/index'
import { observe } from 'core/observer/index'
// 导入内置方法
import {
  warn,
  extend,
  nextTick,
  mergeOptions,
  defineReactive
} from '../util/index'

export function initGlobalAPI (Vue: GlobalAPI) {
  // 定义静态属性
  const configDef = {}
  configDef.get = () => config
  if (process.env.NODE_ENV !== 'production') {
    configDef.set = () => {
      warn(
        'Do not replace the Vue.config object, set individual fields instead.'
      )
    }
  }
  Object.defineProperty(Vue, 'config', configDef) // Vue.config

  // exposed util methods.
  // NOTE: these are not considered part of the public API - avoid relying on
  // them unless you are aware of the risk.
  Vue.util = { // 用不到, 源码会用到 ,必须慎用
    warn,
    extend,
    mergeOptions,
    defineReactive
  }

  Vue.set = set // 设置响应式对象的响应式属性,强制触发视图更新,
  Vue.delete = del // 删除响应式属性强制触发视图更新, 使用情境较少
  Vue.nextTick = nextTick // 结束此轮循环后执行回调,常用于需要等待DOM更新或加载完成后执行的功能

  // 2.6 explicit observable API
  // 对一个新对象添加响应
  Vue.observable = T => {
    observe(obj) // Proxy 
    return obj
  }

  Vue.options = Object.create(null)
  ASSET_TYPES.forEach(type => {
    // Vue.options.components
    // Vue.options.directives
    // Vue.otpions.filters
    Vue.options[type + 's'] = Object.create(null)
  })

  // this is used to identify the "base" constructor to extend all plain-object
  // components with in Weex's multi-instance scenarios.\
  // 在Weex的多实例场景中,它用于标识“基本”构造函数以扩展所有对象组件。  
  Vue.options._base = Vue
  // 扩展全局组件,增加keep-alive、Transition组件
  extend(Vue.options.components, builtInComponents) 

  initUse(Vue) // 初始化Vue.use方法
  initMixin(Vue) // 初始化Vue.mixin
  initExtend(Vue) // 初始化Vue.extend
  initAssetRegisters(Vue) // 初始化Vue.component Vue.filter Vue.directive
}

核心方法Vue.extend

initExtend

// 导入资源类型,模块方法和辅助方法
import { ASSET_TYPES } from 'shared/constants'
import { defineComputed, proxy } from '../instance/state'
import { extend, mergeOptions, validateComponentName } from '../util/index'

// 定义并导出initExtend
export function initExtend (Vue: GlobalAPI) {
  // 每个实例构造函数,包括Vue都有唯一的cid。
  // 这使我们能够为原型继承创建包装的“子构造函数”并缓存它们。
  /**
   * Each instance constructor, including Vue, has a unique
   * cid. This enables us to create wrapped "child
   * constructors" for prototypal inheritance and cache them.
   */
  // 设置Vue的cid为0
  Vue.cid = 0
  // 定义cid变量
  let cid = 1

  // 定义类继承方法
  /**
   * Class inheritance
   */
  // 定义Vue类静态方法extend,接受扩展选项对象
  Vue.extend = function (extendOptions: Object): Function {
    // extendOptions若未定义则设置为空对象
    extendOptions = extendOptions || {}
    // 存储父类和父类的cid
    const Super = this
    const SuperId = Super.cid
    // 定义缓存构造器对象,如果扩展选项的_Ctor属性未定义则赋值空对象
    const cachedCtors = extendOptions._Ctor || (extendOptions._Ctor = {})
    // 如果缓存构造器已存有该构造器,则直接返回
    if (cachedCtors[SuperId]) {
      return cachedCtors[SuperId]
    }

    // 获取扩展配置对象名称或父级配置对象名称属性,赋值给name
    const name = extendOptions.name || Super.options.name
    // 在非生产环境下验证name是否合法并给出警告
    if (process.env.NODE_ENV !== 'production' && name) {
      validateComponentName(name)
    }

    // 定义子类构造函数
    const Sub = function VueComponent (options) {
      this._init(options)
    }
    // 实现子类原型继承,原型指向父类原型,构造器指向Sub
    Sub.prototype = Object.create(Super.prototype)
    Sub.prototype.constructor = Sub
    // 定义子类cid,并递增cid
    Sub.cid = cid++
    // 定义子类options属性,合并配置对象
    Sub.options = mergeOptions(
      Super.options,
      extendOptions
    )
    // 定义子类super属性,指向父类
    Sub['super'] = Super

    // 对于props和computed属性,扩展时在Vue实例上定义了代理getter。
    // 这避免了对每个创建的实例执行Object.defineProperty调用。
    // For props and computed properties, we define the proxy getters on
    // the Vue instances at extension time, on the extended prototype. This
    // avoids Object.defineProperty calls for each instance created.
    // 初始化子类的props
    if (Sub.options.props) {
      initProps(Sub)
    }
    // 初始化子类的计算属性
    if (Sub.options.computed) {
      initComputed(Sub)
    }

    // 定义子类的全局API,扩展、混入和使用插件
    // allow further extension/mixin/plugin usage
    Sub.extend = Super.extend
    Sub.mixin = Super.mixin
    Sub.use = Super.use

    // 创建子类的资源注册方法,允许子类有私有资源
    // create asset registers, so extended classes
    // can have their private assets too.
    ASSET_TYPES.forEach(function (type) {
      Sub[type] = Super[type]
    })
    // 启用递归自查找
    // enable recursive self-lookup
    if (name) {
      Sub.options.components[name] = Sub
    }

    // 在扩展时保持对父类配置对象的引用,
    // 以后实例化时可以检查父级配置对象是否更新
    // keep a reference to the super options at extension time.
    // later at instantiation we can check if Super's options have
    // been updated.
    Sub.superOptions = Super.options
    Sub.extendOptions = extendOptions
    Sub.sealedOptions = extend({}, Sub.options)

    // 缓存子类构造函数
    // cache constructor
    cachedCtors[SuperId] = Sub
    // 返回
    return Sub
  }
}

// 定义初始化propss函数
function initProps (Comp) {
  // 获取配置对象的props属性
  const props = Comp.options.props
  // 设置代理
  for (const key in props) {
    proxy(Comp.prototype, `_props`, key)
  }
}

// 定义初始化计算属性函数
function initComputed (Comp) {
  // 获取配置对象的computed属性
  const computed = Comp.options.computed
  // 设置代理
  for (const key in computed) {
    defineComputed(Comp.prototype, key, computed[key])
  }
}

extend 方法是最为复杂的全局API了,它在扩展类实现继承时进行了很多处理:除去判断是否有已存储的子类构造函数之外,首先是实现类继承,原理是原型式继承;然后为子类初始化props和computed属性的代理:最后是扩展全局API。另外对继承的父类的属性也进行了引用存储。

initUse

// 导入toArray辅助函数
import { toArray } from '../util/index'

// 定义并导出initUse函数
export function initUse (Vue: GlobalAPI) {
  // 定义Vue类静态方法use,接受插件函数或对象
  Vue.use = function (plugin: Function | Object) {
    // 定义内部属性installedPlugins,存放已安装插件
    // 首次应用时定义为空数组
    const installedPlugins = (this._installedPlugins || (this._installedPlugins = []))
    // 检测是否安装过传入的插件,已存在则返回
    if (installedPlugins.indexOf(plugin) > -1) {
      return this
    }

    // 处理附加参数,加入参数Vue
    // additional parameters
    // 将传入的参数转化为数组
    const args = toArray(arguments, 1)
    // 插入Vue类本身为第一个元素
    args.unshift(this)
    // 如果插件有install方法,则在plugin对象上调用并传入新参数
    if (typeof plugin.install === 'function') {
      plugin.install.apply(plugin, args)
    } else if (typeof plugin === 'function') {
      // 如果plugin本身是函数,则直接调用并传入新参数
      plugin.apply(null, args)
    }
    // 向缓存插件数组中添加此插件并返回
    installedPlugins.push(plugin)
    return this
  }
}

initMixin

// 导入mergeOptions辅助函数
import { mergeOptions } from '../util/index'

// 定义并导出initMixin函数
export function initMixin (Vue: GlobalAPI) {
  // 定义Vue的静态方法mixin
  Vue.mixin = function (mixin: Object) {
    // 合并配置对象,重置Vue类的静态属性options
    this.options = mergeOptions(this.options, mixin)
    // 返回
    return this
  }
}

initAssetRegisters

// 导入资源类型和辅助函数
import { ASSET_TYPES } from 'shared/constants'
import { isPlainObject, validateComponentName } from '../util/index'

// 定义并注册initAssetRegisters函数
export function initAssetRegisters (Vue: GlobalAPI) {
  // 创建资源注册方法
  /**
   * Create asset registration methods.
   */
  // 遍历ASSET_TYPES数组,为Vue定义相应方法
  // ASSET_TYPES包括了directive、 component、filter
  ASSET_TYPES.forEach(type => {
    // 定义资源注册方法,参数是标识名称id,和定义函数或对象
    Vue[type] = function (
      id: string,
      definition: Function | Object
    ): Function | Object | void {
      // 如果未传入definition,则视为获取该资源并返回
      if (!definition) {
        return this.options[type + 's'][id]
      } else {
        // 否则视为注册资源
        // 非生产环境下给出检验组件名称的错误警告
        /* istanbul ignore if */
        if (process.env.NODE_ENV !== 'production' && type === 'component') {
          validateComponentName(id)
        }
        // 如果是注册component,并且definition是对象类型
        if (type === 'component' && isPlainObject(definition)) {
          // 设置definition.name属性
          definition.name = definition.name || id
          // 调用Vue.extend扩展定义,并重新赋值
          definition = this.options._base.extend(definition)
        }
        // 如果是注册directive且definition为函数
        if (type === 'directive' && typeof definition === 'function') {
          // 重新定义definition为格式化的对象
          definition = { bind: definition, update: definition }
        }
        // 存储资源并赋值
        this.options[type + 's'][id] = definition
        // 返回definition
        return definition
      }
    }
  })
}

原文地址https://segmentfault.com/a/1190000017359796

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,711评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,079评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,194评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,089评论 1 286
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,197评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,306评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,338评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,119评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,541评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,846评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,014评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,694评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,322评论 3 318
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,026评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,257评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,863评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,895评论 2 351

推荐阅读更多精彩内容