cache-im介绍

背景

原来的Chatroom实例在聊天室组件中,而这次开发的答题器需要接收Chatroom消息。
当时列了几种解决办法:

  1. 把聊天室里面的Chatroom实例暴露出来
  2. Chatroom提供了getInstance方法,可以在答题器组件中取得单例。
  3. 答题器自己实例化一个Chatroom。
  4. 将Chatroom实例从聊天室组件抽出来,做一个分发器,各组件作为订阅者去监听分发器。

第一种办法的问题是:1.组件耦合 2.Chatroom实例化时接收onmsgs回调,不能通过订阅的方式来获取消息,然而两个组件拿到消息之后的处理逻辑又不一样
第二种办法的问题同第一种的第二条,并且需要保证聊天室组件中的Chatroom先实例化。
第三种办法的问题是:需要新的账号,并且发消息的地方也需要。

最终选择了第四种方案。

目标

对于发送方来说:
发消息的时候格式规范化。

对于接收方来说:
组件能够分类接收custom、text、image等云信类别的消息;
组件能够按云信返回消息的顺序接收被派发的消息;
组件可以省去消息判断逻辑,减少冗余消息的干扰:比如开始答题,若没有中间层分发,那么组件会拿到所有的消息自己过滤,去遍历每一个消息,看看type是否为custom且content.eventType是否为开始答题;
省略cache初始化过程;(尚未实现)

术语

(中文-->英文:解释)

云信消息类型-->YXTypes:可以通过msg.type取得,值可能是notification、custom、tip、text、image等。

唯一事件名称-->eventName:用于唯一标识一个事件,如answerSheet#custom_start_answer,即 业务+云信消息类别+具体事件名。

事件类型-->eventType:移动端、前端共用,用于标识主动发送的消息(像系统的notification消息属于被动消息)属于哪种类别,放在msg的content字段中。msgEvent默认根据此来判断是否匹配。

Chatroom:指云信聊天室。

简写

{root} : cache-im/src/im/

设计

消息流图

cache-im介绍_第1张图片

类图

cache-im介绍_第2张图片

流程图

cache-im介绍_第3张图片

使用

需要先在页面中引入云信Chatroom文件,可以在这里下载

发送方

// entry.js
IMCache.getInstance(options);

// component.js
var cache = IMCache.getInstance();
cache.sendCustomMsg({eventType:constant.ANSWER_SHEET_START_ANSWER},callback);   // constant.ANSWER_SHEET_START_ANSWER从{root}constant.js中取得

接收方

回调接收的参数可参考云信文档

// entry.js
IMCache.getInstance(options);

// component.js
var cache = IMCache.getInstance();

cache._$addEvent('msgs',function(msgs){...});

// 即msgs:YXTypes
cache._$addEvent('msgs:custom',function(msgs){...});
cache._$addEvent('msgs:text',function(msgs){...});  

cache._$addEvent('connect',function(chatroom){...});
cache._$addEvent('error',function(error){...});
cache._$addEvent('willConnect',function(obj){...});
cache._$addEvent('disConnect',function(error){...});
cache._$addEvent(xProcessor.CUSTOM.START_ANSWER,function(msgs){...});   // xProcessor在{root}processors/下
... // 具体可参考各processor

:接收到msg.content已经被JSON.parse过

扩展

一、新建processor

在{root}processors创建继承自processor的子类。以下的xxx代表业务范畴,如answerSheet代表答题器,它并不具体指某个组件。

processor主要做的事情就是:定义这个业务范畴需要派发的消息。有个误区需要避免:比如聊天室这个业务范畴,可能其中的聊天室组件需要监听答题器的点赞事件,但是点赞消息并不会归为聊天室的processor中。

/**
 * xxx处理器
 *
 * @version  1.0
 * @author   hzliuzongyuan 
 * @module   pool/cache-im/src/im/processors/xxxProcessor
 */
NEJ.define([
    'base/klass',
    './processor.js',
    'pool/edu-front-util/src/util',
    '../constant.js'
],function(
    k, 
    p,
    util,
    constant,
    exports,
    pro
){
    var xxxProcessor = k._$klass();
    pro = xxxProcessor._$extend(p.Processor);

    pro.__init = function (options) {
        this.prefix = 'xxx';    // 必填,生成eventName的时候需要用到
        this.__super(options,exports);  // 同上
    };

      // custom类消息,每个对象可以填name、eventType、eventName(默认自动生成)、match、handle等属性,具体可以参见{root}msgEvent.js
    pro.customList = function () {
        return [
            {
                name:'connectBrain',    // 用于生成eventName
                eventType:constant.XXX_CONNECT_BRAIN
            }
        ]
    };

      // 类似还有imageList、tipList等,即YXTypes+List,exports.XXX中的XXX即是据此生成的
    pro.textList = function () {
        return [
            {
                name: 'throwDogFood',
                  // 默认是判断this.eventType==msg.content.eventType,可以自己重写
                match: function (msg) {
                    if(msg.text.indexOf('狗粮')!==-1){
                        return true;
                    }
                }
            }
        ]
    };

    // 必须为.Processor,在cache.js中会用到
    exports.Processor = xxxProcessor;
});

二、在{root}list.js中添加processor

三、在{root}constant.js中添加eventType

Q&A

为什么不分发单个消息

单个消息虽然简化了处理逻辑,但可能浪费性能,比如每接收到一条文本则将消息插入到文档中。

为什么运行时才暴露唯一事件名?

如果直接在相应Processor的exports里面暴露,直观是直观了,但是写起来比较麻烦,因为除了上述提到的值之外,还要写key,比如exports.CUSTOM.START_ANSWER。而且手写还可能出错,并且可以不按照规范写,比如直接写成exports.START_ANSWER。
所以这是为了简化人工、规范化。

什么时候才能取得eventName?

processor实例化的时候便会生成eventName,cache会在实例化的时候实例化各processor,也就是说只要先实例化了cache就可以取得了。
一般的使用场景是:

var cache = IMCache.getInstance();
cache.addEvent(XXProcessor.CUSTOM.XXX,function(){});

为什么发送的是eventType,监听的是eventName

首先eventType未必是唯一的,虽然暂时是。现在我们用这个对custom消息进行区分,当以后我们要对notification或者其它云信消息类型也加以区分的时候,按理来说是允许eventType重复的,毕竟可以根据云信消息类型来区分,但是监听方法_$addEvent的第一个参数需要是唯一的。
还有个问题是,如果我们想根据消息本身的内容来新增一个事件时,比如根据text消息内容是否包含「狗粮」,匹配「扔狗粮」事件。如果我们以eventType的形式来接收,那么要给这类事件也分配一个eventType作为标识,而eventType是移动端、前端共用的,这样杂糅在一起不好。

为什么要分processor

我们最简单的分发器就是写一个这样的数组:

[{
    eventType:1,
    handle:function(msg){

    }
},{
    eventType:2,
    handle:function(msg){
        
    }
}...]

然后消息来了,遍历每条消息,看xx字段是否能匹配其中一个项,匹配则调用相应的handle方法,并且派发该消息。

这样我们需要集中管理这个数组,等慢慢积累之后很难维护,分processor其实就是解决这个问题的一种途径。

为什么一个消息只能匹配一个事件

暂时是这样,之后如果遇到相应的需求,也可以放开,不会对之前的逻辑有影响。

为什么不用NIM,而用Chatroom?

NIM包含了好友管理、群组等复杂功能,暂时用不上。

你可能感兴趣的:(cache-im介绍)