Vue3源码之createApp

时间:2025-01-19 07:20:55

前言

Vue3版本源码架构与Vue2版本存在非常大的不同,从使用方式的角度来讲,对它们可以简单概括为:

  • Vue2版本是命令方式使用,通过new Vue去创建Vue实例,组件中使用选项组合代码逻辑
  • Vue3版本将命令和功能API函数化,通过createApp去创建应用实例,组件中通过组合式API和选项组合代码

这里需要区分与Vue2中的一些概念,Vue2中Vue实例实际上就是组件实例,而Vue3中createApp则是创建一个上下文,称之为应用实例。组件实例是通过其他方式来获得的,例如Vue3中mount挂载后就会返回组件实例。(Vue 3.1.1版本)

createApp执行逻辑

在具体梳理createApp之前,聊聊Vue3构建输出的文件结构,不同于Vue2构建输出的UMD规范的文件,Vue3输出的文件主体就是一个IIFE函数:

var Vue = (function(exports) {
 ...
 exports.createApp = createApp;
 ...
})({});

实际上就是输出一个名为Vue的对象,该对象提供createApp函数等。

的执行实际上就是调用一个createApp函数,该函数的逻辑比较清晰,这里直接贴出代码:

const createApp = ((...args) => {
	const app = ensureRenderer().createApp(...args);
 	{
   		injectNativeTagCheck(app);
   		injectCompilerOptionsCheck(app);
 	}
 	const { mount } = app;
 	app.mount = (containerOrSelector) => {
 		// 相关处理
 	}
 	return app;
});

从上面逻辑可知,createApp函数的功能是返回一个app对象,并且会重写mount函数。而创建app对象又会调用其他函数来完成,相关逻辑在后面会详细梳理。

创建Renderer对象

在Vue3中存在Renderer对象的概念,实际上就是渲染器,顾名思义渲染器就是负责节点渲染。

渲染器是延迟创建的,只有在调用ensureRenderer或ensureHydrationRenderer函数才会创建,而且是单例模式。当调用就会调用ensureRenderer函数,其逻辑具体如下:

function ensureRenderer() {
	return renderer ||
		(renderer = createRenderer(rendererOptions));
}

function createRenderer(options) {
  return baseCreateRenderer(options);
}

其中rendererOptions是渲染器相关的参数,具体内容是分为两类:

  • patchProp相关:实际上是组件相关属性、事件等更新相关操作,即patch阶段的处理class、style、on事件绑定等
  • node操作相关的:节点对象的操作,例如插入、移除等

实际上Vue3允许自定义渲染器,对外暴露了createRenderer,在官网文档中有对其的定义描述:

createRenderer 函数接受两个泛型参数: HostNode 和 HostElement,对应于宿主环境中的 Node 和 Element 类型。例如,对于 runtime-dom,HostNode 将是 DOM Node 接口,HostElement 将是 DOM Element 接口。

createRenderer函数背后是调用baseCreateRenderer。

baseCreateRenderer函数

该函数的逻辑简单来说就是返回一个对象,该对象被称为渲染器对象,该对象就3个方法:

function baseCreateRenderer(options, createHydrationFns) {
	// 相关逻辑,其中包含相关patch阶段相关处理的函数定义
	return {
		render,
        hydrate,
        createApp: createAppAPI(render, hydrate)
    };
}

结合Vue2版本的逻辑可以知道,渲染器一部分承担了patch阶段的diff算法的功能和VNode到真实DOM的处理操作。这里先暂时不管这部分的功能,主要看createAppAPI的具体处理逻辑。createAppAPI就是返回一个createApp函数而已:

function createAppAPI(render, hydrate) {
	return function createApp(rootComponent, rootProps = null) {
		// 相关逻辑
	}
}
渲染器的createApp处理

最后就是调用渲染器对象Renderer的createApp函数,其逻辑代码具体如下:

function createApp(rootComponent, rootProps = null) {
   // rootProps必须是对象或null
   if (rootProps != null && !isObject(rootProps)) {
   	warn(`root props passed to () must be an object.`);
    rootProps = null;
   }
   // 创建应用上下文
   const context = createAppContext();
   const installedPlugins = new Set();
   let isMounted = false;
   // 生成具体对象,该对象提供具体的应用API和相关属性
   const app = (context.app = {})
   return app;
}

实际上主要逻辑关注点就两处:

  • createAppContext函数调用创建应用上下文对象
  • 应用对象app的具体内容
调用createAppContext函数创建应用上下文对象

createAppContext函数的具体逻辑如下:

function createAppContext() {
	return {
		// 应用实例
     	app: null,
     	// 应用配置项
        config: {
        	isNativeTag: NO,
            performance: false,
            globalProperties: {},
            optionMergeStrategies: {},
            errorHandler: undefined,
            warnHandler: undefined,
            compilerOptions: {}
         },
         // 混入相关
         mixins: [],
         // 全局组件
         components: {},
         // 指令
         directives: {},
         provides: Object.create(null),
         optionsCache: new WeakMap(),
         propsCache: new WeakMap(),
         emitsCache: new WeakMap()
    };
}
应用实例对象app属性

函数功能就是返回应用实例对象的,该对象处理提供应用API外,还存在一些内部属性:

 const app = (context.app = {
 	_uid: uid$1++,
    _component: rootComponent,
    _props: rootProps,
    _container: null,
    _context: context,
    _instance: null,
    version,
    get config() { return context.config; },
   	set config(v) {
    	{
          warn(` cannot be replaced. Modify individual options instead.`);
       	}
     },
     // 应用API
     use(plugin, ...options) {},
     mixin(mixin) {},
     component(name, component) {},
     directive(name, directive) {},
     mount(rootContainer, isHydrate, isSVG) {},
     unmount() {},
     provide(key, value) {}
});

相关应用API的说明可以具体看Vue3官方文档,而相关内部属性实际上也比较清晰,例如:

  • _context:表示应用上下文实例
  • _component:根组件
  • _container:挂载点容器Node
  • _instance:不出意外应该就是根组件实例对象了
config的额外处理

当生成应用实例对象后,会针对上下文中config配置对象做额外的处理,具体代码如下:

injectNativeTagCheck(app);
injectCompilerOptionsCheck(app);

函数的定义名称是非常清晰的,上面2个函数的具体逻辑实际上就是向config对象中增加:

  • isNativeTag的检查函数:判断是否是HTML标签或SVG标签
  • isCustomElement 和 compilerOptions属性,具体是用来做什么这里暂不关心

总结

是用来创建应用实例的,该应用实例提供相关的应用API和维护相关的状态。在应用实例创建过程中,会创建一个渲染器对象Renderer,针对渲染器对象有如下几点说明:

  • 渲染器对象的创建是惰性的,并且是单例模式
  • Vue3对外暴露了createRenderer API,允许开发者自定义渲染器
  • 渲染器承担Vue3视图渲染相关的功能,其中包含patch阶段的相关处理(diff算法的实现 和 DOM替换等)、应用实例的创建

对比Vue2根组件的处理逻辑(new Vue实际上就是组件实例生成的过程)中data、props、computed等相关处理逻辑,Vue3中相关处理延后到mount挂载阶段。