React 组件通信之发布订阅模式

爱丽丝13
• 阅读 1646

React 通信

react的数据流是单向的, react 通信有以下几种方式:

  • 父向子通信: 传入props
  • 子向父通信:父组件向子组件传一个函数,然后通过这个函数的回调,拿到子组件传过来的值
  • 父向孙通信:利用context传值。React.createContext()
  • 兄弟间通信:

​ 1、找一个相同的父组件,既可以用props传递数据,也可以用context的方式来传递数据。
​ 2、用一些全局机制去实现通信,比如redux等
​ 3、发布订阅模式

兄弟间通信 - 发布订阅模式

组件间通信需要引用一个类的实例,使用单例模式实现。

发布/订阅模式

在 发布/订阅模式 有 发布者订阅者,它们通过信道链接到一起。

其主要包含三个对象:

  • 发布者:消息的发布者,往信道中投递消息的对象。
  • 订阅者:订阅一个或者多个信道消息的对象。
  • 信道:每个信道都有一个名字,信道的实现细节对用户代码来说是隐藏的。

优点

  1. 松耦合:发布者和订阅者的通信是在用户代码之外处理的,通过信道降低了发布者和订阅者的耦合性
  2. 可扩展性:发布/订阅模式可以让系统在无论什么时候都可以扩展
  3. 灵活性:不需要担心不同的组件是如何组合在一起的

缺点

  1. 无法知道消息传送是成功的还是失败的,信道不会通知系统消息传送的状态
  2. 随着订阅者和发布者数量的增加,不断增加的消息传送回导致架构的不稳定,容易在负载大的时候出问题

单例模式

确保一个类仅有一个实例,并提供一个访问它的全局访问点。

代码实现

定义发布对象:

class SingletonPublish {
  constructor() {
    this.listenList = {};
    this.instance = null;
  }

  static getInstance() {
    if (!this.instance) {
      this.instance = new SingletonPublish();
    }
    return this.instance;
  }

  // 订阅者添加订阅事件
  addListen(key, fn) {
    if (!this.listenList[key]) {
      this.listenList[key] = [];
    }
    this.listenList[key].push(fn);
  }

  // 发布者发布消息,执行订阅者订阅事件
  trigger() {
    const key = Array.from(arguments).shift();
    const fns = this.listenList[key];
    if (!fns || fns.length === 0) {
      return false;
    }

    fns.forEach((fn) => {
      fn.apply(this, arguments);
    });
  }

  // 移除订阅事件
  remove(key, fn) {
    const fns = this.listenList[key];
    if (!fns || fns.length === 0) return;

    if (!fn) {
      this.listenList[key] = [];
    } else {
      for (let l = fns.length - 1; l >= 0; l--) {
        if (fn === fns[l]) {
          fns.splice(l, 1);
        }
      }
    }
  }
}

export default SingletonPublish.getInstance();

订阅者订阅一个back事件:

import SingletonPublish from '../singleton-publish';

// ...
SingletonPublish.addListen('back', () => {
  console.log('get -- back');
  SingletonPublish.remove('back', hasExitAndVisible);
});
// ...

发布者发布一个back消息:

import SingletonPublish from '../singleton-publish';

// ...
SingletonPublish.trigger('back');
//... 

观察者模式

在这种模式中,一个目标对象(被观察者)管理所有的依赖于它的对象(观察者),并且在它本身的状态发生变化的时候主动发出通知。

其主要包含两个对象:

  • 被观察者
  • 观察者

缺点

  1. 耦合问题: 每个观察者必须和被观察对象绑定在一起,这引入了耦合
  2. 性能问题:在最基本的实现中观察对象必须同步地通知观察者。这可能会导致性能瓶颈。
点赞
收藏
评论区
推荐文章
浅梦一笑 浅梦一笑
5个月前
初学 Python 需要安装哪些软件?超级实用,小白必看!
编程这个东西是真的奇妙。对于懂得的人来说,会觉得这个工具是多么的好用、有趣,而对于小白来说,就如同大山一样。其实这个都可以理解,大家都是这样过来的。那么接下来就说一下python相关的东西吧,并说一下我对编程的理解。本人也是小白一名,如有不对的地方,还请各位大神指出01名词解释:如果在编程方面接触的比较少,那么对于软件这一块,有几个名词一定要了解,比如开发环
光头强的博客 光头强的博客
5个月前
Java面向对象试题
1、请创建一个Animal动物类,要求有方法eat()方法,方法输出一条语句“吃东西”。创建一个接口A,接口里有一个抽象方法fly()。创建一个Bird类继承Animal类并实现接口A里的方法输出一条有语句“鸟儿飞翔”,重写eat()方法输出一条语句“鸟儿吃虫”。在Test类中向上转型创建b对象,调用eat方法。然后向下转型调用eat()方
Souleigh ✨ Souleigh ✨
2年前
React 组件间通信的10种方法
组件间通信方式总结父组件子组件:1.Props2.InstanceMethods子组件父组件:1.CallbackFunctions2.EventBubbling兄弟组件之间:1.ParentComponent不太相关的组件之间:1.Context2.Portals3.Global
刚刚好 刚刚好
5个月前
css问题
1、在IOS中图片不显示(给图片加了圆角或者img没有父级)<div<imgsrc""/</divdiv{width:20px;height:20px;borderradius:20px;overflow:h
小森森 小森森
5个月前
校园表白墙微信小程序V1.0 SayLove -基于微信云开发-一键快速搭建,开箱即用
后续会继续更新,敬请期待2.0全新版本欢迎添加左边的微信一起探讨!项目地址:(https://www.aliyun.com/activity/daily/bestoffer?userCodesskuuw5n)\2.Bug修复更新日历2.情侣脸功能大家不要使用了,现在阿里云的接口已经要收费了(土豪请随意),\\和注意
马丁路德 马丁路德
1年前
vue3 - 组件通信
vue3组件通信和vue2的通信方式基本一致,只是存在写法上的差异props/emitsetup函数接收两个参数,props和context(上下文,其中有一个emit)用法展示//父组件<template<ibutton:type'type'@onClick
Alex799 Alex799
1年前
vue最新面试题
最近在面试,总结几个重点的面试题:一、vue父子组件之间的传值:简单来说,子组件通过props方法接受父组件传来的值,子组件通过$emit方法来向父组件发送数据。(具体案例可以看我之前写的博客)。二、vue生命周期函数:beforeCreatecreatedbeforeMountmountedbeforeUpdateupdatedbeforeDestr
晴空闲云 晴空闲云
5个月前
css中box-sizing解放盒子实际宽高计算
我们知道传统的盒子模型,如果增加内边距padding和边框border,那么会撑大整个盒子,造成盒子的宽度不好计算,在实务中特别不方便。boxsizing可以设置盒模型的方式,可以很好的设置固定宽高的盒模型。盒子宽高计算假如我们设置如下盒子:宽度和高度均为200px,那么这会这个盒子实际的宽高就都是200px。但是当我们设置这个盒子的边框和内间距的时候,那
艾木酱 艾木酱
5个月前
快速入门|使用MemFire Cloud构建React Native应用程序
MemFireCloud是一款提供云数据库,用户可以创建云数据库,并对数据库进行管理,还可以对数据库进行备份操作。它还提供后端即服务,用户可以在1分钟内新建一个应用,使用自动生成的API和SDK,访问云数据库、对象存储、用户认证与授权等功能,可专
Stella981 Stella981
1年前
React的单向数据流与组件间的沟通
今天来给大家总结下React的单向数据流与组件间的沟通。首先,我认为使用React的最大好处在于:功能组件化,遵守前端可维护的原则。先介绍单向数据流吧。React单向数据流:React是单向数据流,数据主要从父节点传递到子节点(通过props)。如果顶层(父级)的某个props改变了,React会重渲染所有的子节点。刚才我们提到了
helloworld_28799839 helloworld_28799839
5个月前
常用知识整理
Javascript判断对象是否为空jsObject.keys(myObject).length0经常使用的三元运算我们经常遇到处理表格列状态字段如status的时候可以用到vue