我的编程空间,编程开发者的网络收藏夹
学习永远不晚

「项目实战」优化项目构建时间

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

「项目实战」优化项目构建时间

背景

前几天晚上下班的时候, 路过隔壁项目组, 听他们在聊项目构建的事:

现在线上打包时间太长了, 修个 bug 1 分钟, 发布一下半小时, 贼难受。

他们项目比较庞大, 线上构建时间特别长, 基本都在15分钟以上。

和他们简单聊了会, 回去瞅了一下自己项目的构建时间:

其实也挺长的, 于是抽空优化了一下, 效果还是比较明显的。

优化后:

在正文部分,我将分享的内容主要是:

  • 一些提升 webpack 打包性能的配置
  • 优化大型项目构建时间的一些思考

希望对大家有所启发。

 [[399762]]

正文

我们项目不是很大, 是一个中型的国际化项目, 一百来个页面。

之前本地构建时间挺长的,初次启动要三次分钟, 后面我配置了 Vite, 本地启动时间降低到了 20s 左右,感兴趣的可以移步我这篇文章:

[项目实战] Webpack to Vite, 为开发提速!

看了一下,线上构建时间五六分钟,不痛不痒,但是应该也有优化空间,于是准备优化一下。

1. 发现问题

既然要优化构建时间, 第一步当然是先发现问题, 找出比较耗时的阶段,再加以优化。

这里我用到了SMP 插件。

SMP 插件用法非常简单, 这里也简单提一下:

  1. // webpack.config.js 
  2. const SpeedMeasurePlugin = require('speed-measure-webpack-plugin'); 
  3. const smp = new SpeedMeasurePlugin(); 
  4.  
  5. module.exports = smp.wrap({ 
  6.   // ... 
  7. }); 

利用 SMP 插件得出各个阶段的打包耗时:

发现问题

发现两个比较明显的问题:

  1. IgnorePlugin 耗时接近 20 秒。
  2. less-loader 部分执行了2次,浪费了一分多钟。
  3. ts-loader 耗时一分半, 也挺长的。

2. 解决问题

1. IgnorePlugin

查看了一下配置, 发现配置里的 IgnorePlugin 并没有达到预期的效果, 删掉。

2. less-loader

查看配置后发现, 在处理less的部分,确实多处理了一遍。

less 文件的处理,可以直接看官方文档,文档地址:

https://webpack.docschina.org/loaders/less-loader/

我的配置:

  1.   test: /\.less$/, 
  2.   use: [ 
  3.     'style-loader'
  4.     'css-loader'
  5.     { 
  6.       loader: 'less-loader'
  7.       options: { 
  8.         javascriptEnabled: true
  9.         sourceMap: true
  10.         modifyVars: { 
  11.           // inject our own global vars 
  12.           // https://github.com/ant-design/ant-design/issues/16464#issuecomment-491656849 
  13.           hack: `true;@import '${require.resolve('./class="lazy" data-src/vars.less')}';`, 
  14.           ...themeVariables, 
  15.         }, 
  16.         limit: 10000, 
  17.         name'[name].[hash:7].[ext]'
  18.         outputPath: 'styles/'
  19.       }, 
  20.     }, 
  21.   ], 
  22. }, 
  23.   test: /\.css$/, 
  24.   use: ['style-loader''css-loader'], 
  25. }, 

3. ts-loader

对于ts-loader部分的优化, 可以参考:

https://webpack.js.org/guides/build-performance/#typescript-loader

文档上也有比较清晰的描述:

文档建议, 我们开启transpileOnly选项,关闭类型检查。

如果要类型检查, 可以使用 ForkTsCheckerWebpackPlugin,这个插件会在另外一个进程中做相关的检查。

这个插件,我们在优化构建时内存溢出的问题上, 也做了探索, 感兴趣的可以移步我这篇文章:

项目构建内存溢出了?看看 Node 内存限制

现在我们也开启这个选项。

开启之后, 本地构建的时候, 本地报了个警告:

这个错误, 十分的眼熟, 是之前我们讲过的 import type 的问题:

你不知道的 「 import type 」

修复一下: 

问题解决。

重新构建, 得到如下结果:

优化之后之后, 我们发现:

  • IgnorePlugin、HtmlWebpackPlugin 时间大幅缩短。
  • less-loader 等恢复了正常,只执行了一次。
  • ts-loader 时间大幅缩短,由1分30秒缩短为40秒。

本地效果明显,需要去线上构建验证。

3. 确认有效

在线上执行之后, 得到如下结果:

然后去检查了一下页面,也都是正常的。

完美!

回头看,不难发现,其实也没改多少东西, 就收获了不错的效果。

针对中小型项目来说, 改改配置往往就能达到我们的要求, 但是如果是面对大型项目呢?

比如那种数十个模块, 几百个页面的项目。

回到开头那个问题:修个 bug 1 分钟, 发布一下半小时。

简单的修改配置, 都无法把时间降下来, 这时候该怎么办呢?

优化大型项目构建时间的一些思考

拆分子应用

假设我们有一个项目,大模块就有将近30个:

每个大模块里面又有几十个页面,这种系统构建时间会比较久, 需要做优化。

而且到了项目后期,问题会越来越明显, 比如:

  • 代码越来越臃肿
  • 业务模块本身无关联
  • 构建速度越来越慢
  • 无法独立部署

面对这种情况,一种可行的做法是:拆分子应用。

拆分之后的架构:

每个子项目都有单独的入口, 是可以独立部署的项目。

子项目打成单独umd包:

在主项目启动的时候, 再去加载这些子项目:

加载完成之后, 需要处理路由以及store, 示例代码:

  1. // base 
  2. export const bootstrap = () => { 
  3.   // ... 
  4.   ReactDOM.render(( 
  5.      
  6.        
  7.          
  8.        
  9.      
  10.   ), document.getElementById('root')); 
  11.   return Promise.resolve(); 
  12. }; 
  13.  
  14. // main 
  15. const loadSubApp = (htmlEntry: string) => { 
  16.   return importHTML(`${htmlEntry}?${Date.now()}`) 
  17.     .then((res: any) => res.execScripts()) 
  18.     .then((exportedValues: any) => { 
  19.       console.log(`importHTML: ${htmlEntry} loaded, exports:`, exportedValues); 
  20.       const { store, router } = exportedValues || {} as any
  21.       router && addCustomRouter(router); 
  22.       store && addCustomStore(store); 
  23.     }) 
  24.     .catch(e => { 
  25.       console.error('importHTML: ${htmlEntry} load error:', e); 
  26.     }); 
  27. }; 
  28.  
  29. const load = () => { 
  30.   if (__ENV__ !== 'dev') { 
  31.     const paths: string[] = []; 
  32.     subAppConfig.subApps.forEach(item => { 
  33.       if (item.project === localStorage.getItem('ops_project')) { 
  34.         paths.push(...item.paths); 
  35.       } 
  36.     }); 
  37.     Promise.all(paths.map(path => loadSubApp(path))) 
  38.       .catch(e => console.log(e)) 
  39.       .finally(setAllLoaded); 
  40.   } else { 
  41.     setAllLoaded(); 
  42.   } 
  43. }; 
  44.  
  45. const init = () => { 
  46.   console.log('init: Start to bootstrap the main APP'); 
  47.   addCustomStore(rootStore); 
  48.   bootstrap().then(() => { 
  49.     load(); 
  50.   }); 
  51. }; 
  52.  
  53. init(); 

代码共享

  • common包
  • component
  • utils
  • typings
  • ..
  • externals
  • react全家桶
  • moment
  • antd
  • ..

样式隔离

给样式添加以子项目为名的 namespace :

开发调试

以 ops 项目为例。

让开发调试 ops-common 包像本地文件一样方便:

让项目来编译 common 包

wepback alias

TS alias

独立部署

在同一个project上为每个子项目申请独立module

拆分子应用的优缺点

优点:

每个子应用都可以独立发布, 子模块和主模块解耦。

子项目是可以单独编译的,主项目只需要做引入即可, 以此减少主模块的构建时间。

缺点:

额外的复杂性和维护成本

结论

一般来说,对于中小型项目,做好打包配置的优化, 能够解决一部分问题。

大型项目的构建时间优化, 可以考虑拆分子应用的模式。

只不过这种模式需要考虑一些维护的问题,比如如何维护版本 tag、如何快速回滚等。

这些需要结合你们项目的实际情况再做决定。

今天的内容就这么多,希望对大家有所启发。

 

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

「项目实战」优化项目构建时间

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

「项目实战」优化项目构建时间

一般来说,对于中小型项目,做好打包配置的优化, 能够解决一部分问题。大型项目的构建时间优化, 可以考虑拆分子应用的模式。

运维实战项目:Jenkins构建Nodejs前端项目

在Jenkins的CICD过程中,开发人员将代码提交到版本控制系统(如Git)中,Jenkins通过持续监控代码仓库的变化,自动触发构建、测试和部署流程。

项目实战:使用 Fiber + Gorm 构建 Rest API

Fiber 作为一个新的 Go 框架,似乎受追捧程度很高,Star 数飙升很快。不知道这是不是表明,不少 JS/Node 爱好者开始尝试学习 Go 了,对 Go 是好事。
项目Go框架2024-12-02

GO项目实战之Gorm格式化时间字段实现

GORM自带的time.Time类型JSON默认输出RFC3339Nano格式的,下面这篇文章主要给大家介绍了关于GO项目实战之Gorm格式化时间字段实现的相关资料,文中通过实例代码介绍的非常详细,需要的朋友可以参考下
2023-01-07

vite项目构建优化的示例分析

这篇文章主要介绍了vite项目构建优化的示例分析,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。vite项目构建优化路由动态导入 经过rollup的构建,动态导入的文件将会生成
2023-06-20

gomodule构建项目的实现

本文主要介绍了gomodule构建项目的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
2023-03-01

Flask 系列之 优化项目结构

说明操作系统:Windows 10Python 版本:3.7x虚拟环境管理器:virtualenv代码编辑器:VS Code实验目标完善环境配置,添加 异常请求 处理实现400、404 和 500 处理首先,在 todolist\app\t
2023-01-31

Qt创建项目实战之手把手创建第一个Qt项目

我们在进行软件开发学习时,有时候需要qt软件进行代码的敲写,下面这篇文章主要给大家介绍了关于Qt创建项目实战之手把手创建第一个Qt项目的相关资料,文中通过实例代码介绍的非常详细,需要的朋友可以参考下
2023-05-17

PHP互联网项目搭建实战:从零开始搭建完整项目

PHP互联网项目搭建实战:从零开始搭建完整项目随着互联网的不断发展,PHP作为一种常用的后端开发语言,在互联网项目中扮演着重要的角色。本文将带您从零开始,通过具体的代码示例,了解如何搭建一个完整的PHP互联网项目。我们将逐步介绍项目的需求
PHP互联网项目搭建实战:从零开始搭建完整项目
2024-03-08

Vue项目性能优化实践

本文通过以下三部分组成:Vue 代码层面的优化、webpack 配置层面的优化、基础的 Web 技术层面的优化;来介绍怎么去优化 Vue 项目的性能。

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录