简易理解设计模式之:观察者模式——监听与回调
  TEZNKK3IfmPf 10天前 27 0

介绍:

观察者模式属于行为型模式。它定义了一种一对多的依赖关系,让多个观察者对象同时监听某一主题对象。这个主题对象在状态发生变化时,会通知所有观察者对象,使它们能够自动更新自己。

类图:

简易理解设计模式之:观察者模式——监听与回调

Subject(抽象被观察者):抽象主题角色把所有观察者对象的引用保存在一个集合里,并提供可以增加和删除观察者的接口。

ConcreteSubject(具体被观察者):该角色接受具体观察者对象,并在具体主题的内部状态发生改变时,给所有观察者对象发出通知。

Observer(抽象观察者):它定义了一个更新接口,在被观察者发出通知的时候可以更新自己。

ConcreteObserver(具体观察者):实现抽象观察者定义的更新接口,当得到主题更改通知时更新自己。

用法:
• 关联行为场景,当一个对象的改变需要同时改变其它对象
• 跨系统的消息交换场景,如消息队列、事件总线的处理机制

个人理解:
观察者模式应用频率非常高,常用于订阅——发布系统:如通知、广播等业务。还比较常用于GUI系统(图形用户接口),UI层与业务逻辑的解耦。总之,我们在需要使用监听和回调的需求时使用此模式。

例子:
此模式的使用实在太多了,在iOS中的KVC、KVO模式,Android中的BroadCast、View.OnClickListener相信大家已经用过无数遍了。不难发现都是存在监听或者回调的业务需求上,比如在自定义控件想获得它某个处理结果、又或者是统一通知所有界面干一些事情等等。下面就模拟一下这两种比较常用的情况。

需求一:模拟广播的通知
需求二:模拟UI控件的回调

1、实现一个广播
1.1、观察者接口

public interface Receiver {
public void update(String message);
}

抽象的实现可以使抽象类或者是接口。所以我们在观察者接口中定义更新方法,在被观察者发出通知时可以通知自己。

1.2、被观察者接口

public interface Manager {
public void registerObserver(Receiver o);
public void removeObserver(Receiver o);
public void notifyObserver(String message);
}

很简单,在被观察者接口中定义操控观察者的方法。

1.3、广播接受者类(具体观察者)

public class BroadCastReceiver implements Receiver {

private String message;
private String name;

public BroadCastReceiver(String name) {
this.name = name;
}

@Override
public void update(String message) {
this.message = message;
System.out.println(name + message);
}

}

实现了更新接口,接受了广播信息。

1.4、广播管理者(具体被观察者)

public class BroadCastManager implements Manager {

private List<Receiver> mReceivers = new ArrayList<>();

@Override
public void registerObserver(Receiver o) {
mReceivers.add(o);
}

@Override
public void removeObserver(Receiver o) {
mReceivers.remove(o);
}

@Override
public void notifyObserver(String message) {
for (Receiver receiver : mReceivers) {
receiver.update(message);
}
}
}

在这个类中,负责增加和删除广播,向广播发送具体消息。

1.5、使用方法

public static void main(String[] args) {
//观察者
BroadCastReceiver r1 = new BroadCastReceiver("广播一:");
BroadCastReceiver r2 = new BroadCastReceiver("广播二:");


//被观察者
BroadCastManager mManager = new BroadCastManager();
mManager.registerObserver(r1);
mManager.registerObserver(r2);
mManager.notifyObserver("这是一条广播信息");
mManager.removeObserver(r1);
mManager.notifyObserver("测试测试");

}
广播一:这是一条广播信息
广播二:这是一条广播信息
广播二:测试测试

这个例子基本上就完成了,实现了一对多的情况,让两个广播同时监听广播事件。事件监听的代码结构是一种典型的观察者模式结构,下面我们再看一下代码结构上有一点点不一样的回调函数。

2、模拟UI控件回调
2.1、实现一个View类

public class View {

public interface onClickListener {
void onClick(View view);
}

private onClickListener mListener;

public void setOnClickListener(onClickListener listener) {
mListener = listener;
}

public void performClick() {
if (mListener != null) {
mListener.onClick(this);
}
}

}

可能有朋友有疑问:不是说好两个抽象类,两个具体实现类吗,怎么现在都变成一个了?不急,继续往下看。

2.2、实现

public static void main(String[] args) {
View view = new View();
view.setOnClickListener(new View.onClickListener() {
@Override
public void onClick(View view) {
System.out.println(“回调方法”);
}
});
view.performClick();
}

看起来很简化也很熟悉,我们分析一下四个角色:

被观察者:View类(抽象被观察者+具体被观察者)。在setOnClickListener(onClickListener listener)中加入一个观察者对象。
抽象观察者:onClickListener接口,定义回调方法。
具体观察者:new出来的View.onClickListener()对象。

在测试类中,我们模拟控件被执行时调用view.performClick()方法,就会通过回调注册的OnClickListener观察者的onClick方法会来通知观察者,所以回调就是一种观察者模式的具体的实现方式。

总结
观察者模式的具体体现就是回调和监听,另外比较著名的RxJava也是使用此模式,大家也可以看看。

【版权声明】本文内容来自摩杜云社区用户原创、第三方投稿、转载,内容版权归原作者所有。本网站的目的在于传递更多信息,不拥有版权,亦不承担相应法律责任。如果您发现本社区中有涉嫌抄袭的内容,欢迎发送邮件进行举报,并提供相关证据,一经查实,本社区将立刻删除涉嫌侵权内容,举报邮箱: cloudbbs@moduyun.com

  1. 分享:
最后一次编辑于 10天前 0

暂无评论

推荐阅读
  I7JaHrFMuDsU   2024年08月09日   81   0   0 java设计模式
  I7JaHrFMuDsU   2024年08月09日   26   0   0 mvc设计模式
TEZNKK3IfmPf