在vue3中,异步组件可以减少打包的结果,会将异步组件分开打包,会采用异步的方式加载组件,可以有效的解决一个组件过大的问题;不使用异步组件,如果组件功能比较多打包出来的结果就会变大。
本教程操作环境:windows7系统、vue2.9.6版,dell g3电脑。
异步组件是 vue 的一种新能优化的方法,比如可以运用在首屏加载等等场景。
1、异步组件可以减少打包的结果。会将异步组件分开打包,会采用异步的方式加载组件,可以有效的解决一个组件过大的问题。不使用异步组件,如果组件功能比较多打包出来的结果就会变大。
2、异步组件的核心可以给组件定义变成一个函数,函数里面可以用import语法,实现文件的分割加载,import语法是webpack提供的,采用的就是jsonp。
components:{ videoplay:(resolve)=>import("../components/videoplay")} components:{ videoplay(resolve) { require(["../components/videoplay"], resolve) }} 或者使用回调函数
原理在createcomponent方法中,会有相应的异步组件处理,首先定义一个asyncfactory变量,然后进行判断,如果组件是一个函数,然后会去调resolveasynccomponent方法,然后将赋值在asyncfactory上的函数传进去,会让asyncfactory马上执行,执行的时候并不会马上返回结果,因为他是异步的,返回的是一个promise,这时候这个值就是undefined,然后就会先渲染一个异步组件的占位,空虚拟节点。如果加载完之后会调factory函数传入resolve和reject两个参数,执行后返回一个成功的回调和失败的回调,promise成功了就会调resolve,resolve中就会调取forcerender方法强制更新视图重新渲染,forcerender中调取的就是$forceupdate,同时把结果放到factory.resolved上,如果强制刷新的时候就会再次走resolveasynccomponent方法,这时候有个判断,如果有成功的结果就把结果直接放回去,这时候resolveasynccomponent返回的就不是undefined了,就会接的创建组件,初始化组件,渲染组件。
源码src/core/vdom/create-component.js
1、createcomponent方法
export function createcomponent ( ctor: class<component> | function | object | void, data: ?vnodedata, context: component, children: ?array<vnode>, tag?: string): vnode | array<vnode> | void { let asyncfactory if (isundef(ctor.cid)) { // 看组件是否是一个函数 asyncfactory = ctor // 异步组件一定是一个函数 新版本提供了对象的写法 ctor = resolveasynccomponent(asyncfactory, basector) //默认调用此函数时返回undefiend // 第二次渲染时ctor不为undefined if (ctor === undefined) { //返回async组件的占位符节点 //作为注释节点,但保留该节点的所有原始信息 //该信息将用于异步服务器渲染和水合。 return createasyncplaceholder( asyncfactory, data, context, children, tag ) } }}
2、resolveasynccomponent方法
export function resolveasynccomponent ( factory: function, basector: class<component>): class<component> | void { // 如果有错误就返回错误结果 if (istrue(factory.error) && isdef(factory.errorcomp)) { return factory.errorcomp } // 再次渲染时可以拿到获取的最新组件 // 如果有成功的结果,就直接返回去 if (isdef(factory.resolved)) { return factory.resolved } if (owner && !isdef(factory.owners)) { // forcerender 强制刷新渲染 const forcerender = (rendercompleted: boolean) => { for (let i = 0, l = owners.length; i < l; i++) { (owners[i]: any).$forceupdate() // 执行$forceupdate } } // 成功 const resolve = once((res: object | class<component>) => { factory.resolved = ensurector(res, basector) if (!sync) { forcerender(true) // 执行强制更新视图重新渲染方法 } else { owners.length = 0 } }) // 失败 const reject = once(reason => { if (isdef(factory.errorcomp)) { factory.error = true forcerender(true) } }) // 执行factory 将resolve方法和reject方法传入 const res = factory(resolve, reject) sync = false return factory.loading ? factory.loadingcomp : factory.resolved // 返回结果 }}
3、createasyncplaceholder 方法
// 创建一个异步组件的占位,空虚拟节点 也就是一个注释<!-->export function createasyncplaceholder ( factory: function, data: ?vnodedata, context: component, children: ?array<vnode>, tag: ?string): vnode { const node = createemptyvnode() node.asyncfactory = factory node.asyncmeta = { data, context, children, tag } return node}
【相关推荐:《vue.js教程》】
以上就是vue3异步组件有什么用的详细内容。