时间:2021-05-18
本篇文章通过分析react-loadable包的源码,手把手教你实现一个react的异步加载高阶组件
1. 首先我们想象中的react异步加载组件应该如何入参以及暴露哪些API?
通过示例可以看到我们需要入参loaded、loading、delay、timeout,同时暴露单个预加载和全部预加载的API,接下来就让我们试着去一步步实现Loadable高阶组件
2.组件实现过程
整个Loaded函数大体如下
// 收集所有需要异步加载的组件 用于预加载const ALL_INITIALIZERS = [];function Loadable(opts){ return createLoadableComponent(load, opts);}// 静态方法 预加载所有组件Loadable.preloadAll = function(){}接下来实现createLoadableComponent以及load函数
// 预加载单个异步组件function load(loader){ let promise = loader(); let state = { loading: true, loaded: null, error: null, } state.promise = promise.then(loaded => { state.loading = false; state.loaded = loaded; return loaded; }).catch(err => { state.loading = false; state.error = err; throw err; }) return state;}// 创建异步加载高阶组件function createLoadableComponent(loadFn, options){ if (!options.loading) { throw new Error("react-loadable requires a `loading` component"); } let opts = Object.assign({ loader: null, loading: null, delay: 200, timeout: null, }, options); let res = null; function init(){ if(!res){ res = loadFn(options.loader); return res.promise; } } ALL_INITIALIZERS.push(init); return class LoadableComponent extends React{}}我们可以看到createLoadableComponent主要功能包括合并默认配置,将异步组件推入预加载数组,并返回LoadableComponent组件;load函数用于加载单个组件并返回该组件的初始加载状态
接着我们实现核心部分LoadableComponent组件
class LoadableComponent extends React.Component{ constructor(props){ super(props); //组件初始化之前调用init方法下载异步组件 init(); this.state = { error: res.error, postDelay: false, timedOut: false, loading: res.loading, loaded: res.loaded } this._delay = null; this._timeout = null; } componentWillMount(){ //设置开关保证不多次去重新请求异步组件 this._mounted = true; this._loadModule(); } _loadModule(){ if(!res.loading) return; if(typeof opts.delay === 'number'){ if(opts.delay === 0){ this.setState({pastDelay: true}); }else{ this._delay = setTimeout(()=>{ this.setState({pastDelay: true}); }, opts.delay) } } if(typeof opts.timeout === 'number'){ this._timeout = setTimeout(()=>{ this.setState({timedOut: true}); }, opts.timeout) } let update = () => { if(!this._mounted) return; this.setState({ error: res.error, loaded: res.loaded, loading: res.loading, }); } // 接收异步组件的下载结果并重新setState来render res.promise.then(()=>{ update() }).catch(err => { update() }) } // 重新加载异步组件 retry(){ this.setState({ error: null, timedOut: false, loading: false, }); res = loadFn(opts.loader); this._loadModule(); } // 静态方法 单个组件预加载 static preload(){ init() } componentWillUnmount(){ this._mounted = false; clearTimeout(this._delay); clearTimeout(this._timeout); } render(){ const {loading, error, pastDelay, timedOut, loaded} = this.state; if(loading || error){ //异步组件还未下载完成的时候渲染loading组件 return React.createElement(opts.loading, { isLoading: loading, pastDelay: pastDelay, timedOut: timedOut, error: error, retry: this.retry.bind(this), }) }else if(loaded){ // 为何此处不直接用React.createElement? return opts.render(loaded, this.props); }else{ return null; } } }可以看到,初始的时候调用init方法启动异步组件的下载,并在_loadModule方法里面接收异步组件的pending结果,待到异步组件下载完毕,重新setState启动render
接下来还有个细节,异步组件并没有直接启动React.createElement去渲染,而是采用opts.render方法,这是因为webpack打包生成的单独异步组件chunk暴露的是一个对象,其default才是对应的组件
实现如下
function resolve(obj) { return obj && obj.__esModule ? obj.default : obj;} function render(loaded, props) { return React.createElement(resolve(loaded), props);}最后实现全部预加载方法
Loadable.preloadAll = function(){ let promises = []; while(initializers.length){ const init = initializers.pop(); promises.push(init()) } return Promise.all(promises);}整个代码实现如下
const React = require("react");// 收集所有需要异步加载的组件const ALL_INITIALIZERS = [];// 预加载单个异步组件function load(loader){ let promise = loader(); let state = { loading: true, loaded: null, error: null, } state.promise = promise.then(loaded => { state.loading = false; state.loaded = loaded; return loaded; }).catch(err => { state.loading = false; state.error = err; throw err; }) return state;}function resolve(obj) { return obj && obj.__esModule ? obj.default : obj;} function render(loaded, props) { return React.createElement(resolve(loaded), props);}// 创建异步加载高阶组件function createLoadableComponent(loadFn, options){ if (!options.loading) { throw new Error("react-loadable requires a `loading` component"); } let opts = Object.assign({ loader: null, loading: null, delay: 200, timeout: null, render, }, options); let res = null; function init(){ if(!res){ res = loadFn(options.loader); return res.promise; } } ALL_INITIALIZERS.push(init); class LoadableComponent extends React.Component{ constructor(props){ super(props); init(); this.state = { error: res.error, postDelay: false, timedOut: false, loading: res.loading, loaded: res.loaded } this._delay = null; this._timeout = null; } componentWillMount(){ this._mounted = true; this._loadModule(); } _loadModule(){ if(!res.loading) return; if(typeof opts.delay === 'number'){ if(opts.delay === 0){ this.setState({pastDelay: true}); }else{ this._delay = setTimeout(()=>{ this.setState({pastDelay: true}); }, opts.delay) } } if(typeof opts.timeout === 'number'){ this._timeout = setTimeout(()=>{ this.setState({timedOut: true}); }, opts.timeout) } let update = () => { if(!this._mounted) return; this.setState({ error: res.error, loaded: res.loaded, loading: res.loading, }); } res.promise.then(()=>{ update() }).catch(err => { update() }) } // 重新加载异步组件 retry(){ this.setState({ error: null, timedOut: false, loading: false, }); res = loadFn(opts.loader); this._loadModule(); } static preload(){ init() } componentWillUnmount(){ this._mounted = false; clearTimeout(this._delay); clearTimeout(this._timeout); } render(){ const {loading, error, pastDelay, timedOut, loaded} = this.state; if(loading || error){ return React.createElement(opts.loading, { isLoading: loading, pastDelay: pastDelay, timedOut: timedOut, error: error, retry: this.retry.bind(this), }) }else if(loaded){ return opts.render(loaded, this.props); }else{ return null; } } } return LoadableComponent;}function Loadable(opts){ return createLoadableComponent(load, opts);}function flushInitializers(initializers){ }Loadable.preloadAll = function(){ let promises = []; while(initializers.length){ const init = initializers.pop(); promises.push(init()) } return Promise.all(promises);}export default Loadable;到此这篇关于手把手教您实现react异步加载高阶组件的文章就介绍到这了,更多相关react异步加载高阶组件内容请搜索以前的文章或继续浏览下面的相关文章希望大家以后多多支持!
声明:本页内容来源网络,仅供用户参考;我单位不保证亦不表示资料全面及准确无误,也不保证亦不表示这些资料为最新信息,如因任何原因,本网内容或者用户因倚赖本网内容造成任何损失或损害,我单位将不会负任何法律责任。如涉及版权问题,请提交至online#300.cn邮箱联系删除。
怎样才算是一个合格的客服人员?一个好的客服,能够通过自身努力带动起店铺的转化,提升起店铺的销量。客服人员如何实现这一目标,能怎么去实现?点击查看《手把手教您疯狂
手把手教你如何锁定三维视图的方向,方法快捷,好用,是你学习BIM的必须知道的事情!手把手教你如何锁定三维视图的方向软件名称:Autodeskrevit2017中
手把手教您实用技巧:刻录机分位升级...分位:简称Fireware,是刻录机的头脑,它的重要性就不要我们多说了吧,就像主板要经常升级BIOS一样,刻录机的分位同
vue创建高阶组件的实现不够react优雅,但那是vue和react的设计思想导致的。在react中一切都是函数,而在vue中,组件最终都是函数,但在开发时可以
易语言开发windows程序中,经常需要用到超级列表框处理一些数据。那么除了在超级列表框开发中添加数据之外,我们还怎么从TXT中导入内容?下面本教程手把手教您怎