站長資訊網
        最全最豐富的資訊網站

        Angular13+ 開發模式太慢怎么辦?原因與解決方法介紹

        Angular13+ 開發模式太慢怎么辦?下面本篇文章給大家介紹一下Angular 13+ 開發模式太慢的原因與構建性能優化的方法,希望對大家有所幫助!

        Angular13+ 開發模式太慢怎么辦?原因與解決方法介紹

        1 Angular 13+ 開發模式太慢的原因與解決

        近期在某個高頻迭代七年的 Angular 項目升級至 Angular 13 后,其開發模式的構建速度慢、資源占用高,開發體驗相當差。在一臺僅在開會時偶爾使用的 Macbook air(近期居家辦公期間轉換為了主要生產力工具) 中啟動構建時,它的風扇會呼呼作響,CPU 負荷被打滿,而在構建完成后,熱更新一次的時間在一分鐘以上。【相關教程推薦:《angular教程》】

        在經過各種原因分析與排查后,最終在 angular.json 的 schema(./node_modules/@angular/cli/lib/config/schema.json) 中發現了問題,再結合 Angular 12 release 文檔定位到了具體原因: Angular 12 一個主要的改動是將 aotbuildOptimizeroptimization 等參數由默認值 false 改為了 true

        A number of browser and server builder options have had their default values changed. The aim of these changes is to reduce the configuration complexity and support the new "production builds by default" initiative.

        可以看到 Angular 12 后的默認生產模式,對于跨版本升級來說是比較坑爹的。我們可以從這個提交中了解變動細節:656f8d7

        1.1 解決 Angular 12+ 開發模式慢的問題

        解決辦法則是在 development 配置中禁用生產模式相關的配置項。示例:

        {   "$schema": "./node_modules/@angular/cli/lib/config/schema.json",   "projects": {     "front": {       "architect": {         "build": {           "configurations": {             "development": {               "tsConfig": "./tsconfig.dev.json",               "aot": false,               "buildOptimizer": false,               "optimization": false,               "extractLicenses": false,               "sourceMap": true,               "vendorChunk": true,               "namedChunks": true             }           }         },     }   },   "defaultProject": "front" }
        登錄后復制

        需注意 aot 開啟與關閉時,在構建結果表現上可能會有一些差異,需視具體問題而分析。

        1.2 問題:開啟 aotpug 編譯報錯

        該項目中使用 pug 開發 html 內容。關閉 aot 時構建正常,開啟后則會報錯。

        根據報錯內容及位置進行 debugger 調試,可以看到其編譯結果為一個 esModule 的對象。這是由于使用了 raw-loader,其編譯結果默認為 esModule 模式,禁用 esModule 配置項即可。示例(自定義 webpack 配置可參考下文的 dll 配置相關示例):

        {   test: /.pug$/,   use: [     {       loader: 'raw-loader',       options: {         esModule: false,       },     },     {       loader: 'pug-html-loader',       options: {         doctype: 'html',       },     },   ], },
        登錄后復制

        2 進一步優化:Angular 自定義 webpack 配置 dll 支持

        該項目項目構建上有自定義 webpack 配置的需求,使用了 @angular-builders/custom-webpack 庫實現,但是沒有配置 dll。

        Angular 提供了 vendorChunk 參數,開啟它會提取在 package.json 中的依賴等公共資源至獨立 chunk 中,其可以很好的解決熱更新 bundles 過大導致熱更新太慢等的問題,但仍然存在較高的內存占用,而且實際的對比測試中,在存在 webpack5 緩存的情況下,其相比 dll 模式的構建編譯速度以及熱更新速度都稍微慢一些。故對于開發機器性能一般的情況下,給開發模式配置 dll 是會帶來一定的收益的。

        2.1 Angular 支持自定義 webpack 配置

        首先需要配置自定義 webpack 配置的構建支持。執行如下命令添加依賴:

        npm i -D @angular-builders/custom-webpack
        登錄后復制

        修改 angluar.json 配置。內容格式參考:

        {   "$schema": "./node_modules/@angular/cli/lib/config/schema.json",   "cli": {     "analytics": false,     "cache": {       "path": "node_modules/.cache/ng"     }   },   "version": 1,   "newProjectRoot": "projects",   "projects": {     "front": {       "root": "",       "sourceRoot": "src",       "projectType": "application",       "prefix": "app",       "schematics": {         "@schematics/angular:component": {           "style": "less"         }       },       "architect": {         "build": {           "builder": "@angular-builders/custom-webpack:browser",           "options": {             "customWebpackConfig": {               "path": "./webpack.config.js"             },             "indexTransform": "scripts/index-html-transform.js",             "outputHashing": "media",             "deleteOutputPath": true,             "watch": true,             "sourceMap": false,             "outputPath": "dist/dev",             "index": "src/index.html",             "main": "src/app-main.ts",             "polyfills": "src/polyfills.ts",             "tsConfig": "./tsconfig.app.json",             "baseHref": "./",             "assets": [               "src/assets/",               {                 "glob": "**/*",                 "input": "./node_modules/@ant-design/icons-angular/src/inline-svg/",                 "output": "/assets/"               }             ],             "styles": [               "node_modules/angular-tree-component/dist/angular-tree-component.css",               "src/css/index.less"             ],             "scripts": []           },           "configurations": {             "development": {               "tsConfig": "./tsconfig.dev.json",               "buildOptimizer": false,               "optimization": false,               "aot": false,               "extractLicenses": false,               "sourceMap": true,               "vendorChunk": true,               "namedChunks": true,               "scripts": [                 {                   "inject": true,                   "input": "./dist/dll/dll.js",                   "bundleName": "dll_library"                 }               ]             },             "production": {               "outputPath": "dist/prod",               "baseHref": "./",               "watch": false,               "fileReplacements": [                 {                   "replace": "src/environments/environment.ts",                   "with": "src/environments/environment.prod.ts"                 }               ],               "optimization": {                 "scripts": true,                 "styles": {                   "minify": true,                   "inlineCritical": false                 },                 "fonts": true               },               "outputHashing": "all",               "sourceMap": false,               "namedChunks": false,               "aot": true,               "extractLicenses": false,               "vendorChunk": false,               "buildOptimizer": true             }           },           "defaultConfiguration": "production"         },         "serve": {           "builder": "@angular-builders/custom-webpack:dev-server",           "options": {             "browserTarget": "front:build",             "liveReload": false,             "open": false,             "host": "0.0.0.0",             "port": 3002,             "servePath": "/",             "publicHost": "localhost.gf.com.cn",             "proxyConfig": "config/ngcli-proxy-config.js",             "disableHostCheck": true           },           "configurations": {             "production": {               "browserTarget": "front:build:production"             },             "development": {               "browserTarget": "front:build:development"             }           },           "defaultConfiguration": "development"         },         "test": {           "builder": "@angular-builders/custom-webpack:karma",           "options": {             "customWebpackConfig": {               "path": "./webpack.test.config.js"             },             "indexTransform": "scripts/index-html-transform.js",             "main": "src/ngtest.ts",             "polyfills": "src/polyfills.ts",             "tsConfig": "./tsconfig.spec.json",             "karmaConfig": "./karma.conf.js",             "assets": [               "src/assets/",               {                 "glob": "**/*",                 "input": "./node_modules/@ant-design/icons-angular/src/inline-svg/",                 "output": "/assets/"               }             ],             "styles": [               "node_modules/angular-tree-component/dist/angular-tree-component.css",               "src/css/index.less"             ],             "scripts": []           }         }       }     }   },   "defaultProject": "front",   "schematics": {     "@schematics/angular:module": {       "routing": true,       "spec": false     },     "@schematics/angular:component": {       "flat": false,       "inlineStyle": true,       "inlineTemplate": false     }   } }
        登錄后復制

        該示例中涉及多處自定義配置內容,主要需注意 webpack 相關的部分, 其他內容可視自身項目具體情況對比參考。一些細節也可參考以前的這篇文章中的實踐介紹:lzw.me/a/update-to…

        2.2 為 Angular 配置 webpack dll 支持

        新建 webpack.config.js 文件。內容參考:

        const { existsSync } = require('node:fs'); const { resolve } = require('node:path'); const webpack = require('webpack');  // require('events').EventEmitter.defaultMaxListeners = 0;  /**  * @param {import('webpack').Configuration} config  * @param {import('@angular-builders/custom-webpack').CustomWebpackBrowserSchema} options  * @param {import('@angular-builders/custom-webpack').TargetOptions} targetOptions  */ module.exports = (config, options, targetOptions) => {   if (!config.devServer) config.devServer = {};    config.plugins.push(     new webpack.DefinePlugin({ LZWME_DEV: config.mode === 'development' }),   );    const dllDir = resolve(__dirname, './dist/dll');   if (     existsSync(dllDir) &&     config.mode === 'development' &&     options.scripts?.some((d) => d.bundleName === 'dll_library')   ) {     console.log('use dll:', dllDir);     config.plugins.unshift(       new webpack.DllReferencePlugin({         manifest: require(resolve(dllDir, 'dll-manifest.json')),         context: __dirname,       })     );   }    config.module.rules = config.module.rules.filter((d) => {     if (d.test instanceof RegExp) {       // 使用 less,移除 sass/stylus loader       return !(d.test.test('x.sass') || d.test.test('x.scss') || d.test.test('x.styl'));     }     return true;   });    config.module.rules.unshift(     {       test: /.pug$/,       use: [         {           loader: 'raw-loader',           options: {             esModule: false,           },         },         {           loader: 'pug-html-loader',           options: {             doctype: 'html',           },         },       ],     },     {       test: /.html$/,       loader: 'raw-loader',       exclude: [helpers.root('src/index.html')],     },     {       test: /.svg$/,       loader: 'raw-loader',     },     {       test: /.(t|les)s/,       loader: require.resolve('@lzwme/strip-loader'),       exclude: /node_modules/,       options: {         disabled: config.mode !== 'production',       },     }   );    // AngularWebpackPlugin,用于自定義 index.html 處理插件   const awPlugin = config.plugins.find((p) => p.options?.hasOwnProperty('directTemplateLoading'));   if (awPlugin) awPlugin.pluginOptions.directTemplateLoading = false;    // 兼容上古遺傳邏輯,禁用部分插件   config.plugins = config.plugins.filter((plugin) => {     const pluginName = plugin.constructor.name;     if (/CircularDependency|CommonJsUsageWarnPlugin/.test(pluginName)) {       console.log('[webpack][plugin] disabled: ', pluginName);       return false;     }      return true;   });   // console.log('[webpack][config]', config.mode, config, options, targetOptions);   return config; };
        登錄后復制

        新建 webpack.dll.mjs 文件,用于 dll 構建。內容示例:

        import { join } from 'node:path'; import webpack from 'webpack';  const rootDir = process.cwd(); const isDev = process.argv.slice(2).includes('--dev') || process.env.NODE_ENV === 'development';  /** @type {import('webpack').Configuration} */ const config = {   context: rootDir,   mode: isDev ? 'development' : 'production',   entry: {     dll: [       '@angular/common',       '@angular/core',       '@angular/forms',       '@angular/platform-browser',       '@angular/platform-browser-dynamic',       '@angular/router',       '@lzwme/asmd-calc',       // more...     ],   },   output: {     path: join(rootDir, 'dist/dll'),     filename: 'dll.js',     library: '[name]_library',   },   plugins: [     new webpack.DllPlugin({       path: join(rootDir, 'dist/dll/[name]-manifest.json'),       name: '[name]_library',     }),     new webpack.IgnorePlugin({       resourceRegExp: /^./locale$/,       contextRegExp: /moment$/,     }),   ],   cache: { type: 'filesystem' }, };  webpack(config).run((err, result) => {   console.log(err ? `Failed!` : `Success!`, err || `${result.endTime - result.startTime}ms`); });
        登錄后復制

        angular.json 中添加 dll.js 文件的注入配置,可參考前文示例中 development.scripts 中的配置內容格式。

        package.json 中增加啟動腳本配置。示例:

        {     "scripts": {         "ng:serve": "node --max_old_space_size=8192 node_modules/@angular/cli/bin/ng serve",         "dll": "node config/webpack.dll.mjs",         "dev": "npm run dll -- --dev && npm run ng:serve -- -c development",     } }
        登錄后復制

        最后,可執行 npm run dev 測試效果是否符合預期。

        3 小結

        angular-cli 在升級至 webpack 5 以后,基于 webpack 5 的緩存能力做了許多編譯優化,一般情況下開發模式二次構建速度相比之前會有大幅的提升。但是相比 snowpackvite 一類的 esm no bundles 方案仍有較大的差距。其從 Angular 13 開始已經在嘗試引入 esbuild,但由于其高度定制化的構建邏輯適配等問題,對一些配置參數的兼容支持相對較為復雜。在 Angular 15 中已經可以進行生產級配置嘗試了,有興趣也可作升級配置與嘗試。

        贊(0)
        分享到: 更多 (0)
        網站地圖   滬ICP備18035694號-2    滬公網安備31011702889846號
        主站蜘蛛池模板: 久久精品国产99久久丝袜| 亚洲AV无码精品无码麻豆| 欧美精品人爱a欧美精品| 97久久综合精品久久久综合| 自拍偷自拍亚洲精品第1页| 国产精品成人99久久久久91gav| 国产精品对白交换视频| 亚洲日韩精品射精日 | 亚洲欧洲国产精品香蕉网| 国产激情精品一区二区三区| 国产精品成人va| 国内精品久久久久伊人av| 亚洲国产小视频精品久久久三级 | 无码国产亚洲日韩国精品视频一区二区三区 | 亚洲综合一区二区国产精品| 国产精品视频永久免费播放| 日韩精品无码一区二区三区免费| 天天爽夜夜爽8888视频精品| 久久夜色撩人精品国产小说| 久久久精品国产亚洲成人满18免费网站 | 国产精品免费一区二区三区| 国产精品免费观看| 9久久9久久精品| 国精品午夜福利视频不卡| 无码精品日韩中文字幕| 亚洲愉拍99热成人精品热久久| 亚洲第一区精品日韩在线播放| 久久99国产精品成人欧美| 精品国产毛片一区二区无码| 国精品无码A区一区二区| 精品国产午夜福利在线观看| 精品久久久久久无码人妻蜜桃 | 青草青草久热精品视频在线网站 | 亚洲婷婷国产精品电影人久久| 中文字幕精品无码一区二区 | 亚洲一区精品中文字幕| 国产成人精品一区二三区在线观看| 亚洲国产精品久久久久久| 一区二区三区国产精品| 国产精品hd免费观看| 国产精品亚洲欧美大片在线观看|