微前端如何落地?

在过去的几星期里,随着 Martin Fowler 博客上,那篇 Cam Jackson 写的微前端的文章发布,处处都在讨论 Microfrontend。做为一个微前端 “专家”,我也分享一下:如何去落地微前端。前端

微前端是一种相似于微服务的架构,它将微服务的理念应用于浏览器端,即将单页面前端应用由单一的单体应用转变为多个小型前端应用聚合为一的应用。各个前端应用还能够独立开发、独立部署。同时,它们也能够在共享组件的同时进行并行开发——这些组件能够经过 NPM 或者 Git Tag、Git Submodule 来管理。web

为何须要微前端?

微前端不是银弹,它和微服务同样会带来大量的挑战。数据库

  • 遗留系统迁移。解决遗留系统,才是人们采用微前端方案最重要的缘由
  • 聚合前端应用。微服务架构,能够解耦后端服务间依赖。而微前端,则关注于聚合前端应用。
  • 热闹驱动开发。新的技术,既然很热闹,那么就学吧。

微前端的实现,意味着对前端应用的拆分。拆分应用的目的,并不仅是为了架构上好看,还为了提高开发效率。bootstrap

为此,微前端带来这么一系列的好处:后端

  1. 应用自治。只须要遵循统一的接口规范或者框架,以便于系统集成到一块儿,相互之间是不存在依赖关系的。
  2. 单一职责。每一个前端应用能够只关注于本身所须要完成的功能。
  3. 技术栈无关。你可使用 Angular 的同时,又可使用 React 和 Vue。

除此,它也有一系列的缺点:浏览器

  • 应用的拆分基础依赖于基础设施的构建,一旦大量应用依赖于同一基础设施,那么维护变成了一个挑战。
  • 拆分的粒度越小,便意味着架构变得复杂、维护成本变高。
  • 技术栈一旦多样化,便意味着技术栈混乱

毕竟没有银弹。前端框架

如何设计微前端架构?

就当前而言,要设计出一个微前端应用不是一件容易的事——尚未最佳实践。在不一样的落地案例里,使用的都是不一样的方案。出现这种状况的主要缘由是,每一个项目所面临的状况、所使用的技术都不尽相同。为此,咱们须要了解一些基础的微前端模式。服务器

架构模式

微前端应用间的关系来看,分为两种:基座模式(管理式)、自组织式。分别也对应了二者不一样的架构模式:markdown

  • 基座模式。经过一个主应用,来管理其它应用。设计难度小,方便实践,可是通用度低。
  • 自组织模式。应用之间是平等的,不存在相互管理的模式。设计难度大,不方便实施,可是通用度高。

就当前而言,基座模式实施起来比较方便,方案上便也是蛮多的。架构

而不论种方式,都须要提供一个查找应用的机制,在微前端中称为服务的注册表模式。和微服务架构类似,不管是哪一种微前端方式,也都须要有一个应用注册表的服务,它能够是一个固定值的配置文件,如 JSON 文件,又或者是一个可动态更新的配置,又或者是一种动态的服务。它主要作这么一些内容:

  • 应用发现。让主应用能够寻找到其它应用。
  • 应用注册。即提供新的微前端应用,向应用注册表注册的功能。
  • 第三方应用注册。即让第三方应用,能够接入到系统中。
  • 访问权限等相关配置。

应用在部署的时候,即可以在注册表服务中注册。若是是基于注册表来管理应用,那么使用基座模式来开发比较方便。

设计理念

在笔者实践微前端的过程当中,发现了如下几点是咱们在设计的过程当中,须要关注的内容:

  • 中心化:应用注册表。这个应用注册表拥有每一个应用及对应的入口。在前端领域里,入口的直接表现形式能够是路由,又或者对应的应用映射。
  • 标识化应用。 咱们须要一个标识符来标识不一样的应用,以便于在安装、卸载的时候,能寻找到指定的应用。一个简单的模式,就是经过康威定律来命名应用。
  • 应用生命周期管理。
  • 高内聚,低耦合。

生命周期

前端微架构与后端微架构的最大不一样之处,也在于此——生命周期。微前端应用做为一个客户端应用,每一个应用都拥有本身的生命周期:

  • Load,决定加载哪一个应用,并绑定生命周期
  • bootstrap,获取静态资源
  • Mount,安装应用,如建立 DOM 节点
  • Unload,删除应用的生命周期
  • Unmount,卸载应用,如删除 DOM 节点、取消事件绑定

这部分的内容事实上,也就是微前端的一个难点所在,如何以合适的方式来加载应用——毕竟每一个前端框架都各自不一样,其所须要的加载方式也是不一样的。当咱们决定支持多个框架的时候,便须要在这一部分进入更细致的研究。

如何拆分?

随后,咱们要面临的一个挑战是:如何去拆分应用。

技术方式

从技术实践上,微前端架构能够采用如下的几种方式进行:

  • 路由分发式。经过 HTTP 服务器的反向代理功能,来将请求路由到对应的应用上。
  • 前端微服务化。在不一样的框架之上设计通信、加载机制,以在一个页面内加载对应的应用。
  • 微应用。经过软件工程的方式,在部署构建环境中,组合多个独立应用成一个单体应用。
  • 微件化。开发一个新的构建系统,将部分业务功能构建成一个独立的 chunk 代码,使用时只须要远程加载便可。
  • 前端容器化。经过将 iFrame 做为容器,来容纳其它前端应用。
  • 应用组件化。借助于 Web Components 技术,来构建跨框架的前端应用。

实施的方式虽然多,可是都是依据场景而采用的。有些场景下,可能没有合适的方式;有些场景下,则能够同时使用多种方案。

路由分发式

**路由分发式微前端,即经过路由将不一样的业务分发到不一样的、独立前端应用上。**其一般能够经过 HTTP 服务器的反向代理来实现,又或者是应用框架自带的路由来解决。以下图所示:

路由分发式

就当前而言,路由分发式的架构应该是采用最多、最容易的 “微前端” 方案。可是

前端微服务化

**前端微服务化,是微服务架构在前端的实施,每一个前端应用都是彻底独立(技术栈、开发、部署、构建独立)、自主运行的,最后经过模块化的方式组合出完整的前端应用。**其架构以下图所示:

路由分发式

采用这种方式意味着,一个页面上同时存在二个及以上的前端应用在运行。而路由分发式方案,则是一个页面只有惟一一个应用。

组合式集成:微应用化

**微应用化,即在开发时,应用都是以单1、微小应用的形式存在,而在运行时,则经过构建系统合并这些应用,组合成一个新的应用。**其架构以下图所示:

路由分发式

微应用化更多的是以软件工程的方式,来完成前端应用的开发,所以又能够称之为组合式集成。对于一个大型的前端应用来讲,采用的架构方式,每每会是经过业务做为主目录,然后在业务目录中放置相关的组件,同时拥有一些通用的共享模板。

微件化

**微件(widget),指的是一段能够直接嵌入在应用上运行的代码,它由开发人员预先编译好,在加载时不须要再作任何修改或者编译。**而微前端下的微件化则指的是,每一个业务团队编写本身的业务代码,并将编译好的代码部署(上传或者放置)到指定的服务器上,在运行时,咱们只须要加载相应的业务模块便可。对应的,在更新代码的时候,咱们只须要更新对应的模块便可。下图即是微件化的架构示意图:

路由分发式

在非单面应用时代,要实现微件化方案,是一件特别容易的事。从远程加载来对应的 JavaScript 代码,在浏览器上执行,生成对应的组件嵌入到页面的相应部分。对于业务组件也是相似的,提早编写好咱们的业务组件,当须要对应的组件时再响应、执行。

前端容器化

前端容器:iframe

iframe 做为一个很是 “古老” 的,人人都以为普通的技术,却一直很管用。它能有效地将另外一个网页/单页面应用嵌入到当前页面中,两个页面间的 CSS 和 JavaScript 是相互隔离的——除去 iframe 父子通讯部分的代码,它们之间的代码是彻底不相互干扰的。iframe 便至关因而建立了一个全新的独立的宿主环境,相似于沙箱隔离,它意味着前端应用之间能够相互独立运行。

结合 Web Components 构建

Web Components 是一套不一样的技术,容许开发者建立可重用的定制元素(它们的功能封装在代码以外),而且在 web 应用中使用它们。

路由分发式

目前困扰 Web Components 技术推广的主要因素,在于浏览器的支持程度。在 Chrome 和 Opera 浏览器上,对于 Web Components 支持良好,而对于 Safari、IE、Firefox 浏览器的支持程度,并无那么理想。

业务拆分

与微服务相似,要划分不一样的前端边界,不是一件容易的事。就当前而言,如下几种方式是常见的划分微前端的方式:

  • 按照业务拆分
  • 按照权限拆分
  • 按照变动的频率拆分
  • 按照组织结构拆分。利用康威定律来进一步拆分前端应用。
  • 跟随后端微服务划分。实践证实, DDD 与事件风暴是一种颇为有效的后端微前端拆分模式,对于前端来讲,它也很有有效——直接跟踪后端服务。

每一个项目都有本身特殊的背景,切分微前端的方式便不同。即便项目的类型类似,也存在一些细微的差别。

微前端以外

若是微前端对于大家来讲困境重重,还有一些不错的架构模式能够采用。

应用微化架构

应用微化架构,是一种开发时总体,构建时拆分,运行时分离的前端架构模式。即应用微化架构从一份代码中,构建出适用于不一样环境的多套目标代码。实现上它是一种:

  • 构建时拆分架构
  • 代码删除架构。笑,以删代码的方式,来造成每一个前端应用。
  • 微前端准备式架构。即,随时能够拆分为多个前端应用。

因为它与微应用化的类似性,咱们将它与微应用化作一个对比。它与微应用化不一样的是,应用微化是在构建时对应用进行拆分,而非在本地模式下对应用拆分。类似的是,它也是基于构建系统的应用拆分方式。

可拆分式微前端

即:微应用化,是一个随时可合并式架构。而应用微化,则是一个随时可拆分式架构

它不只仅是一个适合于前端的架构模式,也是一适用于后端的架构模式

整洁前端架构

Clean Architecture 是由 Robert C. Martin 在 2012 年提出的架构模式。它具备这么一些特色:框架无关性、可被测试、UI 无关性、数据库无关性、外部机构(agency)无关性。

对于前端架构来讲,Clean Architecure 其实是:Clean Architecture + MVP + 组件化。以下图所示:

考虑到应用的规模,这里以 Angular + TypeScript 做为示例:

Clean Frontend

这种架构模式特别适合于:组织内即写前端又同后端的团队。它易于映射先后端 API,且可使用 usecase 做为防腐层。

没有银弹。不得不说起的是,对于小规模的团队来讲,它带来的弊端可能会远远大于好处——带来大量冗余的代码。尽管经过 Angular Schematics 能够经过参数来生成代码,可是这种分层架构地于简单的应用来讲,仍是过于复杂、难以上手。对于不写测试的项目来讲 ,usecase 也不能带来它们所承诺的好处。

结论

微前端不是银弹,当前也没有最佳实践,可是这是一个很是好的学习机会。


节选自《前端架构:从入门到微前端》