Vue 2 阅读理解(九)之 mergeOptions 配置合并

简介: Vue 2 阅读理解(九)之 mergeOptions 配置合并

1. mergeOptions 配置合并


这里就省略内部组件的配置合并,只是几个简单属性的合并,源码位于 src/core/util/options.ts


const strats = config.optionMergeStrategies
const defaultStrat = function (parentVal: any, childVal: any): any {
  return childVal === undefined ? parentVal : childVal
}
export function mergeOptions(parent: Record<string, any>,child: Record<string, any>,vm?: Component | null): ComponentOptions {
  if (__DEV__) {
    checkComponents(child)
  }
  if (isFunction(child)) {
    child = child.options
  }
  normalizeProps(child, vm)
  normalizeInject(child, vm)
  normalizeDirectives(child)
  if (!child._base) {
    if (child.extends) {
      parent = mergeOptions(parent, child.extends, vm)
    }
    if (child.mixins) {
      for (let i = 0, l = child.mixins.length; i < l; i++) {
        parent = mergeOptions(parent, child.mixins[i], vm)
      }
    }
  }
  const options: ComponentOptions = {} as any
  let key
  for (key in parent) {
    mergeField(key)
  }
  for (key in child) {
    if (!hasOwn(parent, key)) {
      mergeField(key)
    }
  }
  function mergeField(key: any) {
    const strat = strats[key] || defaultStrat
    options[key] = strat(parent[key], child[key], vm, key)
  }
  return options
}


整个过程首先校验了 child 中的组件命名配置,并且如果传入的 child 是一个函数的话,则直接获取 child.options 重新赋值给 child  作为后面的合并项。


然后,会格式化 childprops, inject, directives 三项配置,将其处理为标准的 对象格式


之后,则是遍历 childextendsmixins 配置,依次调用 mergeOptions 方法,将所有配置全部合并到 parent 中。


最后,则是遍历 parentchild,将所有的配置项组合到一个新的配置项对象 options 中并返回。


这里遍历 child 在 遍历 parent 之后,所以最终的配置项都是 以 Child 配置为主,不存在的配置则使用 Parent 的配置项


parent 与 child 之间的同配置项合并可以通过在 Vue.config.optionMergeStrategies 中配置对应的处理函数来配置自定义合并。


2. defaultStrats 默认合并策略


在该模块的顶部,定义了一个空对象 strats = config.optionMergeStrategies 用来保存内部配置的合并策略,并在后面重新定义每个配置项的合并策略,避免默认的 data,methods 类型等配置的合并策略被覆盖。


const strats = config.optionMergeStrategies
const defaultStrat = function (parentVal: any, childVal: any): any {
  return childVal === undefined ? parentVal : childVal
}
if (__DEV__) {
  strats.el = strats.propsData = function (parent: any, child: any, vm: any, key: any) {
  }
}
strats.data = function (parentVal: any, childVal: any, vm?: Component): Function | null {
}
LIFECYCLE_HOOKS.forEach(hook => {
  strats[hook] = mergeLifecycleHook
})
ASSET_TYPES.forEach(function (type) {
  strats[type + 's'] = mergeAssets
})
strats.watch = function (
  parentVal: Record<string, any> | null,
  childVal: Record<string, any> | null,
  vm: Component | null,
  key: string
): Object | null {
}
strats.props =
  strats.methods =
  strats.inject =
  strats.computed =
    function (parentVal: Object | null, childVal: Object | null, vm: Component | null, key: string): Object | null {}
strats.provide = mergeDataOrFn


以上部分分别定义了以下几个部分的合并策略:


  • mergeData:会遍历 parent 的 data 对象 (如果是函数的话,会重新赋值为函数的返回值),并区分以下情况


  1. parent 中有 child 中没有的属性,则将该属性进行响应式处理并赋给 child


       b. parent 与 child 都有但是两者的值不相等,且都是对象的时候,会继续遍历两个对象执行合并方法


       c. 排除以上情况,则都以 child 中的配置为准


  • mergeLifecycleHook :生命周期合并策略,会将组件内的生命周期钩子函数配置转为数组格式,然后将所有的生命周期配置函数合并到一个数组中


  • mergeAssets:资源合并策略,主要合并 components,directives,filter 配置,一样是以 child 配置覆盖 parent 配置


  • mergeWatch:合并 watch 监听配置,与生命周期合并类似,同名属性监听会将监听函数放置到数组中顺序执行


  • mergeOther:合并 props,inject,methods 等配置,与资源合并策略类似,同名配置以 child 为准


  • mergeProvide:合并注入数据,采用与 mergeData 一样的合并策略


在定义 strats 之后还定义了 strats.elstrats.propsData 的合并策略,但是内部使用的还是 defaultStrat,只是增加了一层校验,警告只能在 new Vue 的时候使用定义这些配置。


最终,在标准化处理和合并完成之后的配置,大概就长这样:


options = {
  el: '#app',
  filters: {},
  props: {
    propOne: { type: [String, Number], default: 0 }
  },
  directives: {}, // 原型链上会有默认的 model 和 show
  components: {   // 原型链上会有默认的 Translation,TranslationGroup,KeepAlive
    subComponent: { components: {}, name: '', /** */ },
  },
  data() {},
  created: [createdFc1, createdFc2],
  mounted: [mountedFc],
  watch: {
    xxx1: function() {},
    xxx2: {
      deep: true,
      immediate: true,
      handler: function() {}
    }
  },
  _base: Vue,
  // 使用 Vue.component 之类的方式注册的组件会有 _Ctor
  _Ctor: [
    VueComponent
  ]
}


目录
相关文章
|
2月前
|
JavaScript 前端开发
如何在 Vue 项目中配置 Tree Shaking?
通过以上针对 Webpack 或 Rollup 的配置方法,就可以在 Vue 项目中有效地启用 Tree Shaking,从而优化项目的打包体积,提高项目的性能和加载速度。在实际配置过程中,需要根据项目的具体情况和需求,对配置进行适当的调整和优化。
|
3月前
|
JavaScript 数据可视化
vue-cli学习二:vue-cli3版本 创建vue项目后,Runtime-Compiler和Runtime-only两个模式详解;vue项目管理器;配置文件的配置在哪,以及如何配置
这篇文章详细介绍了Vue CLI 3版本创建项目时的Runtime-Compiler和Runtime-only两种模式的区别、Vue程序的运行过程、render函数的使用、eslint的关闭方法,以及Vue CLI 2和3版本配置文件的不同和脚手架3版本创建项目的配置文件配置方法。
224 3
vue-cli学习二:vue-cli3版本 创建vue项目后,Runtime-Compiler和Runtime-only两个模式详解;vue项目管理器;配置文件的配置在哪,以及如何配置
|
2月前
|
JavaScript 前端开发 开发者
如何在 Visual Studio Code (VSCode) 中使用 ESLint 和 Prettier 检查代码规范并自动格式化 Vue.js 代码,包括安装插件、配置 ESLint 和 Prettier 以及 VSCode 设置的具体步骤
随着前端开发技术的快速发展,代码规范和格式化工具变得尤为重要。本文介绍了如何在 Visual Studio Code (VSCode) 中使用 ESLint 和 Prettier 检查代码规范并自动格式化 Vue.js 代码,包括安装插件、配置 ESLint 和 Prettier 以及 VSCode 设置的具体步骤。通过这些工具,可以显著提升编码效率和代码质量。
654 4
|
3月前
|
资源调度 JavaScript 前端开发
路由管理:Vue Router的使用和配置技巧
【10月更文挑战第21天】路由管理:Vue Router的使用和配置技巧
64 3
|
3月前
|
存储 前端开发 中间件
vue3之vite配置vite-plugin-mock使用mock轻松创建模拟数据提高开发效率
vue3之vite配置vite-plugin-mock使用mock轻松创建模拟数据提高开发效率
693 0
|
4月前
|
JavaScript
vue项目中使用vue-router进行路由配置及嵌套多级路由
该文章详细说明了如何在Vue项目中配置和使用vue-router进行单页面应用的路由管理,包括设置嵌套路由和实现多级路由导航的示例。
|
4月前
|
JavaScript
Vue3基础(20)___Vue3配置错误路由重定向写法
本文介绍了Vue 3中配置错误路由重定向的正确写法,包括使用参数和自定义正则表达式来定义通配符路由。
70 0
Vue3基础(20)___Vue3配置错误路由重定向写法
|
3月前
|
JavaScript
vue尚品汇商城项目-day01【2.vue-cli脚手架初始化项目的其他配置】
vue尚品汇商城项目-day01【2.vue-cli脚手架初始化项目的其他配置】
38 0
|
5月前
|
JavaScript
基于Vue2或Vue3实现任意上下左右拖拽悬浮的元素,且配置为自定义的全局指令
这篇文章介绍了如何在Vue 2或Vue 3项目中实现一个自定义的全局指令`v-dragSwitch`,用于创建可以任意方向拖拽并悬浮的元素,同时包含边界处理的逻辑。
1595 2
基于Vue2或Vue3实现任意上下左右拖拽悬浮的元素,且配置为自定义的全局指令
|
5月前
|
JavaScript
在Vue中使用Avue、配置过程以及实际应用
这篇文章介绍了作者在Vue项目中使用Avue组件库的体验,包括安装配置过程和实际应用示例,展示了如何通过Avue实现动态增加输入框和输入验证的功能。
在Vue中使用Avue、配置过程以及实际应用