基于qiankun的微前端最佳实践-通信篇(Vuex)

大家好~~

在开始介绍 qiankun 的应用通信之前,我们需要先了解微前端架构如何划分子应用。

在微前端架构中,我们应该按业务划分出对应的子应用,而不是通过功能模块划分子应用。这么做的原因有两个:

  1. 在微前端架构中,子应用并不是一个模块,而是一个独立的应用,我们将子应用按业务划分可以拥有更好的可维护性和解耦性。
  2. 子应用应该具备独立运行的能力,应用间频繁的通信会增加应用的复杂度和耦合度。

综上所述,我们应该从业务的角度出发划分各个子应用,尽可能减少应用间的通信,从而简化整个应用,使得我们的微前端架构可以更加灵活可控。

上面的这段话是引自这篇好文,这篇文章中主要介绍了两种主应用和微应用的通信方式

第一种是 qiankun 官方提供的通信方式 - Actions 通信,适合业务划分清晰,比较简单的微前端应用,一般来说使用第一种方案就可以满足大部分的应用场景需求。

第二种是基于 redux 实现的通信方式 - Shared 通信,适合需要跟踪通信状态,子应用具备独立运行能力,较为复杂的微前端应用。

我写这篇博客的主要原因是探究主应用和多个微应用之间如何使用Vuex来进行状态管理,这其实是对于上面通信方式中,第二种方式的补充。

场景

项目现状:

  1. 主应用是Vue技术栈,使用Vuex进行状态管理
  2. 多个微应用也是Vue技术栈,并且都可能使用Vuex进行状态管理

想要解决的问题:

  1. 主应用与微应用的通信
  2. 微应用之间的通信

通信方式的选择:

  1. qiankun 官方提供的通信方式 - Actions 通信。

    优点:

    • 使用简单;
    • 官方支持性高;
    • 适合通信较少的业务场景;

    缺点:

    • 子应用独立运行时,需要额外配置无 Actions 时的逻辑;
    • 子应用需要先了解状态池的细节,再进行通信;
    • 由于状态池无法跟踪,通信场景较多时,容易出现状态混乱、维护困难等问题;
  2. 使用Vuex进行状态管理

    优点:

    • 子应用无法随意污染主应用的状态池,只能通过主应用暴露的 shared 实例的特定方法操作状态池,从而避免状态池污染产生的问题。
    • 子应用将具备独立运行的能力

    缺点:

    暂时没有想到。。。。

核心思想

Vuex通信Store共享.jpg

主要就是在做微前端集成时,将主应用的store共享给所有微应用,针对不同微应用可以进行模块化设置,这样使得共享的state更加易于管理。

具体实现(微应用原本就没有使用Vuex进行状态管理)

step1:主应用向微应用传递store实例。

registerMicroApps(
      [
          {
              name: "chai-project",
              entry: "//localhost:8080",
              container: '#yourContainer',
              activeRule: "/chaiQiankunTest/ffff",
              props: {
                store //共享主应用的store实例
              }
          }
      ],
      {
          beforeLoad: [
              app => {
                  console.log("before load", app);
              }
          ], // 挂载前回调
          beforeMount: [
              app => {
                  console.log("before mount", app);
              }
          ], // 挂载后回调
          afterUnmount: [
              app => {
                  console.log("after unload", app);
              }
          ] // 卸载后回调
      }
    )

step2:微应用使用主应用共享的store实例

针对第一种情况,就是在入口文件中引入vuex,并使用该插件,进而在创建vue实例的时候,传入主应用共享的store实例。

import Vuex from 'vuex'
Vue.use(Vuex);
function render (props) {
  const store = props.store;
  // 在 render 中创建 VueRouter,可以保证在卸载微应用时,移除 location 事件监听,防止事件污染
  router = new Router({
    // 运行在主应用中时,添加路由命名空间 /chaiQiankunTest/ffff
    base: window.__POWERED_BY_QIANKUN__ ? 'chaiQiankunTest/ffff' : '/',
    mode: 'history',
    routes
  });

  // 挂载应用
  instance = new Vue({
    router,
    store,//主应用共享的store实例
    render: (h) => h(App)
  }).$mount('#app');
}

step3:验证主应用和微应用之间是否可以完成通信,状态同步。

验证的思路就是

(1)在主应用和微应用的页面上添加改变同一个state的方法,并且利用computed实时显示该state的状态值。

(具体的实现代码这里就不详细给出)

验证用例主要是

(1)点击主应用按钮,修改state值之后,主应用和微应用的页面上都实时触发computed属性,展示最新状态值

(2)点击微应用按钮,修改state值之后,主应用和微应用的页面上都实时触发computed属性,展示最新状态值

验证的结果

不论是点击主应用的按钮,还是点击微应用的按钮,主应用的computed属性成功被触发,微应用始终未能正常监听到状态值得改变,computed属性从未被触发。

这里查阅了蛮多的资料,没能看到有大神对qiankun使用Vuex的经验总结(这里也发现自己有点过于依赖网络搬砖,有些问题积极思考是能够找到答案的)

step4:bug修复

微应用的computed始终没有被触发,底层原因就是其依赖属性this.$store.state.count是非相应式的,这会导致难以触发。

computed: {
    childCount: {
      get (){
        return this.$store.state.count;
      }
    }
}

Vuex正常使用的时候,所有的状态值都是响应式的,可以直接用于Vue页面之中,但是这里是非响应式的,导致这个的原因其实十分简单。这里的store实例是由主应用传递过来的,store中的状态对于主应用的Vue实例而言是亲儿子,是响应式的,在微应用中,虽然可以使用共享store实例中的commit方法,但是对于微应用的Vue实例而言,不是亲儿子,是非响应式的,这样分析之后,解决方案就十分明确:

在微应用中将共享的store实例进行响应式设置,这是Vue现有的API方法Vue.observable(store)

如此处理之后,不论主应用还是微应用中,修改共享storestate状态值,在主应用和微应用中都能够实时感知,并对其做出响应的反馈。

具体实现(微应用本身就有自己store实例)

这种情况主要考虑的是两种方案

  1. 主应用需要提前得知子应用的store内容,将其与主应用的store进行某种融合,这样子应用在集成环境中以及在独立运行时都能够正常运行。

    缺点:这方方案会导致主应用和子应用相互耦合过强,并且主应用和子应用都要维护一份子应用的store,增加工作量,过于笨重。

  2. 主应用依旧只传递主应用的store,子应用的store依旧在使用,也就是在子应用中既可以操作主应用的store,也可以操作子应用的store

    优点:主应用只需关注需要交互的状态即可,不用关心子应用原本的store内容。子应用也只需关注自己的store内容即可,降低耦合,减轻复杂度。

父子应用store分离的方案实现

这里基于父应用已经共享自己的store,并且主应用和子应用之间已经能够完成对于主应用的state状态变化的响应。考虑如何在子应用中使用两个storenew Vue()中目前注入的是主应用的store,那子应用本身的store如何全局注册呢?仔细思考后,其实很简单,只需要在入口文件中添加如下一行代码:

Vue.prototype.microStore = microStore;

如此一来,在子应用的各个页面都能够通过this.microStore访问自身的store

computed: {
    microCount: {
        get (){
            return this.microStore.state.microCount;
        }
    }
}

总结

到此,关于在qiankun的集成中,主应用和子应用使用Vuex进行通信的基本使用方法介绍结束了,缺少了子应用和子应用之间的一个实验,大家有兴趣可以试试。

参考

  1. 基于 qiankun 的微前端最佳实践(图文并茂) - 应用间通信篇
  2. 目标是最完善的微前端解决方案 - qiankun 2.0
  3. qiankun官网API指导
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,163评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,301评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,089评论 0 352
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,093评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,110评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,079评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,005评论 3 417
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,840评论 0 273
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,278评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,497评论 2 332
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,667评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,394评论 5 343
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,980评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,628评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,796评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,649评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,548评论 2 352