跳转到主要内容

标签(标签)

资源精选(342) Go开发(108) Go语言(103) Go(99) angular(83) LLM(79) 大语言模型(63) 人工智能(53) 前端开发(50) LangChain(43) golang(43) 机器学习(39) Go工程师(38) Go程序员(38) Go开发者(36) React(34) Go基础(29) Python(24) Vue(23) Web开发(20) Web技术(19) 精选资源(19) 深度学习(19) Java(18) ChatGTP(17) Cookie(16) android(16) 前端框架(13) JavaScript(13) Next.js(12) 安卓(11) 聊天机器人(10) typescript(10) 资料精选(10) NLP(10) 第三方Cookie(9) Redwoodjs(9) ChatGPT(9) LLMOps(9) Go语言中级开发(9) 自然语言处理(9) PostgreSQL(9) 区块链(9) mlops(9) 安全(9) 全栈开发(8) OpenAI(8) Linux(8) AI(8) GraphQL(8) iOS(8) 软件架构(7) RAG(7) Go语言高级开发(7) AWS(7) C++(7) 数据科学(7) 智能体(6) whisper(6) Prisma(6) 隐私保护(6) JSON(6) DevOps(6) 数据可视化(6) wasm(6) 计算机视觉(6) 算法(6) Rust(6) 微服务(6) 隐私沙盒(5) FedCM(5) 语音识别(5) Angular开发(5) 快速应用开发(5) 提示工程(5) Agent(5) LLaMA(5) 低代码开发(5) Go测试(5) gorm(5) REST API(5) kafka(5) 推荐系统(5) WebAssembly(5) GameDev(5) CMS(5) CSS(5) machine-learning(5) 机器人(5) 游戏开发(5) Blockchain(5) Web安全(5) nextjs(5) Kotlin(5) 低代码平台(5) 机器学习资源(5) Go资源(5) Nodejs(5) PHP(5) Swift(5) RAG架构(4) devin(4) Blitz(4) javascript框架(4) Redwood(4) GDPR(4) 生成式人工智能(4) Angular16(4) Alpaca(4) 编程语言(4) SAML(4) JWT(4) JSON处理(4) Go并发(4) 移动开发(4) 移动应用(4) security(4) 隐私(4) spring-boot(4) 物联网(4) 网络安全(4) API(4) Ruby(4) 信息安全(4) flutter(4) 专家智能体(3) Chrome(3) CHIPS(3) 3PC(3) SSE(3) 人工智能软件工程师(3) LLM Agent(3) Remix(3) Ubuntu(3) GPT4All(3) 软件开发(3) 问答系统(3) 开发工具(3) 最佳实践(3) RxJS(3) SSR(3) Node.js(3) Dolly(3) 移动应用开发(3) 低代码(3) IAM(3) Web框架(3) CORS(3) 基准测试(3) Go语言数据库开发(3) Oauth2(3) 并发(3) 主题(3) Theme(3) earth(3) nginx(3) 软件工程(3) azure(3) keycloak(3) 生产力工具(3) gpt3(3) 工作流(3) C(3) jupyter(3) 认证(3) prometheus(3) GAN(3) Spring(3) 逆向工程(3) 应用安全(3) Docker(3) Django(3) R(3) .NET(3) 大数据(3) Hacking(3) 渗透测试(3) C++资源(3) Mac(3) 微信小程序(3) Python资源(3) JHipster(3) 语言模型(2) 可穿戴设备(2) JDK(2) SQL(2) Apache(2) Hashicorp Vault(2) Spring Cloud Vault(2) Go语言Web开发(2) Go测试工程师(2) WebSocket(2) 容器化(2) AES(2) 加密(2) 输入验证(2) ORM(2) Fiber(2) Postgres(2) Gorilla Mux(2) Go数据库开发(2) 模块(2) 泛型(2) 指针(2) HTTP(2) PostgreSQL开发(2) Vault(2) K8s(2) Spring boot(2) R语言(2) 深度学习资源(2) 半监督学习(2) semi-supervised-learning(2) architecture(2) 普罗米修斯(2) 嵌入模型(2) productivity(2) 编码(2) Qt(2) 前端(2) Rust语言(2) NeRF(2) 神经辐射场(2) 元宇宙(2) CPP(2) 数据分析(2) spark(2) 流处理(2) Ionic(2) 人体姿势估计(2) human-pose-estimation(2) 视频处理(2) deep-learning(2) kotlin语言(2) kotlin开发(2) burp(2) Chatbot(2) npm(2) quantum(2) OCR(2) 游戏(2) game(2) 内容管理系统(2) MySQL(2) python-books(2) pentest(2) opengl(2) IDE(2) 漏洞赏金(2) Web(2) 知识图谱(2) PyTorch(2) 数据库(2) reverse-engineering(2) 数据工程(2) swift开发(2) rest(2) robotics(2) ios-animation(2) 知识蒸馏(2) 安卓开发(2) nestjs(2) solidity(2) 爬虫(2) 面试(2) 容器(2) C++精选(2) 人工智能资源(2) Machine Learning(2) 备忘单(2) 编程书籍(2) angular资源(2) 速查表(2) cheatsheets(2) SecOps(2) mlops资源(2) R资源(2) DDD(2) 架构设计模式(2) 量化(2) Hacking资源(2) 强化学习(2) flask(2) 设计(2) 性能(2) Sysadmin(2) 系统管理员(2) Java资源(2) 机器学习精选(2) android资源(2) android-UI(2) Mac资源(2) iOS资源(2) Vue资源(2) flutter资源(2) JavaScript精选(2) JavaScript资源(2) Rust开发(2) deeplearning(2) RAD(2)

I have work with Angular framework in 3 software companies. In the first days at each company, my task was to read the source code then all 3 times I had to say “What the hell with SharedModule”. They declared shared components, pipes, directives, Material UI module then import SharedModule where feature module need one of them, that’s too bad. So why?

Noted: From Angular 14, we have standard alone that has same idea with SCAM I will mention bellow.

@NgModule({
  declarations: [...components, ...pipes, ...directives],
  imports: [
    ...modules,
    MaterialModule
  ],
  exports: [...modules, ...components, ...pipes, ...directives],
})
export class SharedModule {}

Problems

Let’s make an example for case using SharedModule.

We will make an example with a project Angular use shared module. We have a shared module with 2 components: sharedA and sharedB. We also have 2 lazy-loaded modules feature-a and feature-b module. feature-a use sharedA component, feature-b use sharedB component, then we import SharedModule into 2 feature modules to it can use shared component.

@NgModule({
  declarations: [
    SharedBComponent,
    SharedAComponent,
  ],
  exports: [
    SharedAComponent,
    SharedBComponent
  ]
})
export class SharedModule { }

------------------------------
@NgModule({
  declarations: [FeatureBComponent],
  imports: [RouterModule.forChild(routes), SharedModule]
})
export class FeatureBModule {}

---------------------------------

@NgModule({
  declarations: [FeatureAComponent],
  imports: [RouterModule.forChild(routes), SharedModule]
})
export class FeatureAModule {}

Now we use webpack-bundle-analyzer to see chunks by Angular CLI when building

Visual overview of our generated bundles with Shared Module

With 2 lazy-loaded modules ( feature-afeature-b), that’s mean Angular CLI will make 2 chunks for these modules. But you can see, SharedModule also has a distinct chunk with 2 feature modules. Oh yes, it make lazy-loaded modules don’t increase bundle size, but when browser load these modules, it also load Shared Module.

It’s easy to see that this is bad practice. feature-a only use sharedA component but it needs to load the entire SharedModule including sharedB component. When SharedModule grow bigger day by day, it will become a heavy dumbbell in project. So what is solution for this?

Solution

One module per component (SCAM) is our solution.

  • Remove Shared Module
  • Create SharedA module for sharedA component, SharedB Module for sharedB component. This apply for directive, pipe,… one thing one module
  • Feature-a module import SharedA module (because Feature-a need sharedA component), same for Feature-b
Visual overview of our generated bundles after apply SCAM

You can see, Shared Module chunk is not created, only chunks of lazy-loaded modules. Bundle size of lazy-loaded module increase a bit because it will include sharedA module, but look at the overview, now each lazy-loaded module only load anything necessary for it, browser will not carrying a heavy weight Shared Module anymore.

Some people asked me what happens with 3rd library that imported in lazy-loaded modules?

My answers is with 3rd library such as HttpClientModule, FormModule,…let import it on lazy-loaded module needs. If you import it only one lazy-loaded module, 3rd library will be loaded on chunk of that lazy-loaded feature

Bundle size when I import HttpClientModule on feature-b module

But if you import 3rd library in multiple lazy-loaded modules, Angular CLI will realize this library used in many module then there’s a chunk to load this 3rd library

Bundle size when I import HttpClientModule on both feature modules

Conclusion

By splitting modules, we can build feature modules with what it needs. SCAM (Single component as module) is a best pattern with its benefits that I mention above, reducing bundle size, easy for maintaining and refactoring code. If your project has Shared Module, it’s time for refactoring your code. Good luck!!!