Android 本身也是有观察者模式的。虽然项目中很多需要通知数据改变的地方,用了EventBus,但是不得不说这个观察者模式还是很好用的。最近在开发新版本的时候引用了腾讯的IM,之前写直播的时候就用了,当时只在TCChatRoomMgr中注册了消息的监听,因为当时只在直播中用了收发消息,且是单一的聊天室。但是项目的聊天整体接入IM后这个问题就变得棘手了,因为很多界面要接收消息,包括首页,推送,聊天列表,还有直播,这该怎么办呢?模仿腾讯聊天IM的demo,demo中用的就是观察者模式,即只注册一次消息监听(腾讯的文档的代码),然后在所有需要观察的地方注册观察者,根据消息类型进行判断和过滤,问题就迎刃而解了。

观察者模式:简单来说,就是当对象A对对象B进行进行了类似“订阅”关系,当对象B的数据发生改变时,就要通知对象A进行相应。很简单也很好理解。Android中的观察者需要实现Observer接口,当数据发生改变时,观察者的update()方法就会被调用,被观察者继承Observable类,在数据发生改变时,需要调用setChanged(); this.notifyObservers(obj);这两个方法才可以通知观察者:你想要知道的数据发生了变化了。好了,废话不多说了,下面直接上代码,我都在代码里头做了注释了,各位看官请直接看下面的代码。

注册观察者:

public class MessageEvent extends Observable implements TIMMessageListener {

    private volatile static MessageEvent instance;

    private MessageEvent(){
//注册消息监听器
TIMManager.getInstance().addMessageListener(this);
} public static MessageEvent getInstance(){
if (instance == null) {
synchronized (MessageEvent.class) {
if (instance == null) {
instance = new MessageEvent();
}
}
}
return instance;
} @Override
public boolean onNewMessages(List<TIMMessage> list) {
for (TIMMessage item:list){
setChanged();
notifyObservers(item);
}
return false;
} /**
* 主动通知新消息
*/
public void onNewMessage(TIMMessage message){
setChanged();
notifyObservers(message);
} /**
* 清理消息监听
*/
public void clear(){
instance = null;
}
}

  在任何想要接收消息的地方注册观察这就可以了比如ActivityHome:

public class ActivityHome extends ThinksnsAbscractActivity implements OnChatListener, UnreadMessageListener, OnClickListener, Observer {...
 //在登陆后调用这个
MessageEvent.getInstance().addObserver(ActivityHome.this);
 @Override
public void update(Observable observable, Object data) {
if (observable instanceof MessageEvent) {
TIMMessage msg = (TIMMessage) data;
//系统消息,自己发的消息,程序在前台的时候不通知
if (msg == null || Foreground.get().isForeground() ||
(msg.getConversation().getType() != TIMConversationType.Group &&
msg.getConversation().getType() != TIMConversationType.C2C) ||
msg.isSelf() ||
msg.getRecvFlag() == TIMGroupReceiveMessageOpt.ReceiveNotNotify ||
(MessageFactory.getMessage(msg) instanceof CustomMessage &&
((CustomMessage) MessageFactory.getMessage(msg)).getType() == CustomMessage.Type.TYPING))
return;
int im_unread_num = PreferenceUtils.getInt(Thinksns.getMy().getUid()+"im_unread_num", 0);
if (msg.getConversation().getType() == TIMConversationType.C2C) {
PreferenceUtils.putInt(Thinksns.getMy().getUid()+"im_unread_num", ++im_unread_num);
if (mdNotification != null) {
setUnReadUi(mdNotification);
} else {
fg_my.setUnReadMsg(0);
}
} else if(msg.getConversation().getType() == TIMConversationType.Group && !msg.getConversation().getPeer().contains("@")){
PreferenceUtils.putInt(Thinksns.getMy().getUid()+"im_unread_num", ++im_unread_num);
if (mdNotification != null) {
setUnReadUi(mdNotification);
} else {
fg_my.setUnReadMsg(0);
}
}
// ToastUtils.t("123456789");
}
}

再比如聊天列表页的Presenter里:

public class ConversationPresenter implements Observer {

    private static final String TAG = "ConversationPresenter";
private ConversationView view; public ConversationPresenter(ConversationView view){
    
//注册消息监听
MessageEvent.getInstance().addObserver(this);
//注册刷新监听
RefreshEvent.getInstance().addObserver(this);
//注册好友关系链监听
FriendshipEvent.getInstance().addObserver(this);
//注册群关系监听
GroupEvent.getInstance().addObserver(this);
this.view = view;
}
  //在这个地方处理消息即可
@Override
public void update(Observable observable, Object data) {
if (observable instanceof MessageEvent){
TIMMessage msg = (TIMMessage) data;
view.updateMessage(msg);
}else if (observable instanceof FriendshipEvent){
FriendshipEvent.NotifyCmd cmd = (FriendshipEvent.NotifyCmd) data;
switch (cmd.type){
case ADD_REQ:
case READ_MSG:
case ADD:
view.updateFriendshipMessage();
break;
}
}else if (observable instanceof GroupEvent){
GroupEvent.NotifyCmd cmd = (GroupEvent.NotifyCmd) data;
switch (cmd.type){
case UPDATE:
case ADD:
view.updateGroupInfo((TIMGroupCacheInfo) cmd.data);
break;
case DEL:
view.removeConversation((String) cmd.data);
break; }
}else if (observable instanceof RefreshEvent){
view.refresh();
}
}
...
}

   使用了这种方法后,上面的困扰就解决了,注册abserver的地方还包括推送,直播页面,注意直播页面的TCChatroomMgr里面收消息一定要进行过滤后判断会话是否为空,因为这里就只收相应聊天室的消息,如果正常操作,退出直播后会退出聊天室,聊天室的消息就不收了,除非遇到闪退的情况(极少),用户还在这个聊天室中,那只能等到聊天解散了,其他对应界面也过滤了直播聊天室的消息。主要是开发起来比较麻烦,张杰磊同学辛苦了!!

Created by WangXiaotao

Android之观察者/被观察者模式Observer/Observable的更多相关文章

  1. java: 观察者模式:Observable被观察者,Observer观察者

    java: 观察者模式:Observable被观察者,Observer观察者 以房子价格为例,卖房者为被观察者: import java.util.Observable; //被观察者子类化 publ ...

  2. [Android&amp;Java]浅谈设计模式-代码篇:观察者模式Observer

    观察者,就如同一个人,对非常多东西都感兴趣,就好像音乐.电子产品.Game.股票等,这些东西的变化都能引起爱好者们的注意并时刻关注他们.在代码中.我们也有这种一种方式来设计一些好玩的思想来.今天就写个 ...

  3. Java_观察者模式(Observable和Observer) -转

    原文地址: Java_观察者模式(Observable和Observer) 一.观察者模式介绍    在Java中通过Observable类和Observer接口实现了观察者模式.一个Observer ...

  4. 十一、观察者模式(Observable、Observer)

    老板出差了,员工1.员工2..均放羊中.他们请求前台的秘书,当老板回来时通知自己,免得被Boss抓个现行.秘书想了想,说————嗯,这是观察者模式. 当一个对象的改变需要同时改变其它对象,而且它不知道 ...

  5. android中的所谓观察者模式

    生活中我们常认定某些人很有才,但什么是有才呢?明朝的王守仁曾这样解释:才,是所谓天理,应用到物上,便成了才.凡事凡物,只要掌握了所谓科学的方法,并能灵活运用,那么你也可以成为一个有才的人. 观察者模式 ...

  6. C# ~ 从 委托事件 到 观察者模式 - Observer

    委托和事件的部分基础知识可参见 C#/.NET 基础学习 之 [委托-事件] 部分: 参考 [1]. 初识事件 到 自定义事件: [2]. 从类型不安全的委托 到 类型安全的事件: [3]. 函数指针 ...

  7. [设计模式] 19 观察者模式 Observer Pattern

    在GOF的<设计模式:可复用面向对象软件的基础>一书中对观察者模式是这样说的:定义对象间的一种一对多的依赖关系,当一个对象的状态发生改变时,所有依赖于它的对象都得到通知并被自动更新.当一个 ...

  8. 我理解设计模式C++实现观察者模式Observer Pattern

    概述: 近期中国股市起起伏伏,当然了起伏就用商机,小明发现商机后果断想入市,买入了中国证券,他想在电脑client上,网页上,手机上,iPad上都能够查看到该证券的实时行情,这样的情况下我们应该怎么设 ...

  9. java设计模式--观察者模式(Observer)

    java设计模式--观察者模式(Observer) java设计模式--观察者模式(Observer) 观察者模式的定义: 定义对象间的一种一对多的依赖关系.当一个对象的状态发生改变时,所有依赖于它的 ...

随机推荐

  1. Windows 10 建立wifi热点

    如果当前是台式机那么需要一个usb的无线网卡,这里要注意如果你是使用台式机并且通过有线的方式上网,但是你的无线网卡适配器不能在禁用状态. 这里首先打开[运行]输入cmd,打开cmd(注意,这里要使用管 ...

  2. XML 转 fastJSON

      import java.util.List; import org.dom4j.Attribute; import org.dom4j.Document; import org.dom4j.Doc ...

  3. webpack4搭建Vue开发环境笔记~~持续更新

    项目git地址 一.node知识 __dirname: 获取当前文件所在路径,等同于path.dirname(__filename) console.log(__dirname); // Prints ...

  4. PHP安装Xcache扩展

    简述 XCache 是一个又快又稳定的 ​PHP opcode 缓存器. 经过良好的测试并在大流量/高负载的生产机器上稳定运行. 经过(在 linux 上)测试并支持所有现行 ​PHP 分支的最新发布 ...

  5. visual studio 2013 for windows desk报error MSB8020: The build tools for v141错误

    由于硬件限制,学习在touchgfx暂时在Windows下模拟仿真,了解其基本原理和有一个基本感性认识,因此安装了VS Express 2013 for Desktop轻量级编译器. 有TouchGF ...

  6. 关于host,nslookup,dig 的安装

    host,nslookup,dig依赖bind包,所以先看一下系统有没有bind包 命令如下:rpm -qa |grep bind 如果没有或者版本太低请升级安装 命令是:yum install bi ...

  7. UVa 10110 Light, more light

    开始所有的灯是灭的,不过我们只关心最后一个灯. 在第i次走动时,只有编号为i的倍数的灯的状态才会改变. 也就是说n有偶数个约数的时候,最后一个灯的状态不会改变,也就是灭的. n有奇数个约数的时候也就是 ...

  8. ogre3D学习基础11 -- 交换两个场景管理器

    这一节,练习一下前几次学习的内容,功能很简单,就是建立两个不同的场景管理器,当按下键盘上某个键时切换镜头. 基本框架不变,这个监听器继承了两个父类,一个是我们的老朋友ExampleFrameListe ...

  9. 大数据学习——scala入门练习

    package com /** * Created by ZX on 2015/11/6. */ object VariableDemo { def main(args: Array[String]) ...

  10. hibernate缓存机制【转】

    一.why(为什么要用Hibernate缓存?) Hibernate是一个持久层框架,经常访问物理数据库. 为了降低应用程序对物理数据源访问的频次,从而提高应用程序的运行性能. 缓存内的数据是对物理数 ...