观察者模式(新手推荐)

今天给大家带来一个较为简单的模式,观察者模式。如果觉得我写得还不错,记得关注下,我好有勇气给大家以浅显的语言介绍完这几种设计模式。

为什么要使用观察者模式?

举个简单的例子,在一所工科学校里(我们都知道,工科院校女生都比较少),有一个很有教养,漂亮,温柔的女生大家都很喜欢,自然有很多人追。女生的一举一动,大家都很关注。比如女生半夜发了个状态,说“我饿了”。

。。。。。。

接下来,不得了了,众男们(假设现在有A,B,C三个男生同时喜欢上了改女生)着急了,都想用自己的方法让女生填饱肚子(A:走,带你吃KFC。B:别动,我给你买了送到你楼下。C:自己画个饼,看看就行了)。当然以C的这种方式,肯定没戏,咳咳,扯远了。那么用伪代码如何表示呢?

abstract class Boy{

//照顾自己身边的人是每个男生义不容辞的责任
pulbic void careDarling();

}

// A 照顾人的方式
class BoyA extends Boy{

pulbic void careDarling(){
    
    System.out.println("走,带你吃KFC");
}

}

//B 照顾人的方式

class BoyB extends Boy{

pulbic void careDarling(){
    
    System.out.println("别动,我给你买了送到你楼下");
}

}

//C 照顾人的方式

class BoyC extends Boy{

pulbic void careDarling(){
    
    System.out.println("自己画个饼,看看就行了");
}

}

//下面这个是女孩类
class Girl{


public void hungrey(){
    
    //饿了,我们要通知A,B,C 是时候行动了    
    BoyA.careDarling();
    BoyB.careDarling();
    BoyC.careDarling();

}

}

当然,我们尊重每个人爱人的方式。现在不是纠结这个的时候,我们看上面伪代码的表示有什么问题吗?

就在下看,有发现几个问题:

  1. 作为女孩,首先我要知道谁喜欢我啊,不然通知错人了怎么办?(实际情况是,尽管你知道有人喜欢你,但是这个人是谁你知道吗?)
  2. 又有人喜欢女孩了,也想照顾她。但是不想让女孩知道,只想默默地付出。按照我们现在这种写法,只有女孩知道谁喜欢她才会通知到,不知道的就不通知了。
  3. 其他

使用观察者模式有什么好处

看了我们上面的写法,发现这种写法是不合适的。我们有没有解决办法呢?
答案当然是有啊。但是首先我们应明确自己的需求。

  1. 我只要主动关注女孩就行了,不用等通知,自己要主动,毕竟幸福靠自己争取。
  2. 女孩魅力大,我们要允许其他人也喜欢(不限于A,B,C),毕竟每个人都有爱与被爱的权利。

所以,观察者模式来救场。

观察者模式结构

(声明下,我有空了就去学UML。这个画的不够标准,希望能说明问题。)

那么对应于我们这个例子,之间关系又是怎样的呢?

新的结构

那我们为何不用代码实现下呢?

Observer

public interface Observer {

public void update();
}

Boy

public abstract class Boy {

public abstract void careDarling();
}

BoyA

public class BoyA extends Boy implements Observer {

private Subject subject;
public BoyA(Subject subject){
    subject.addObserver(this);
}

@Override
public void update() {
    careDarling();
}

@Override
public void careDarling() {
    System.out.println("走,带你吃KFC");
}
}

BoyB

public class BoyB extends Boy implements Observer {

private Subject subject;
public BoyB(Subject subject){
    subject.addObserver(this);
}
@Override
public void update() {
    careDarling();
}

@Override
public void careDarling() {
    System.out.println("别动,我给你买了送到你楼下");
}
}

BoyC

public class BoyC extends Boy implements Observer {

private Subject subject;
public BoyC(Subject subject){
    subject.addObserver(this);
}

@Override
public void update() {
    careDarling();
}

@Override
public void careDarling() {
    System.out.println("自己画个饼,看看就行了");
}

}

Subject

public interface Subject {

public void addObserver(Observer boy);
public void removeObserver(Observer boy);
public void notifyObservers();    
}

Girl

public class Girl implements Subject {

private List<Observer> boys;

public Girl(){
    boys = new ArrayList<Observer>();
}

@Override
public void addObserver(Observer boy) {
    
    boys.add(boy);

}

@Override
public void removeObserver(Observer boy) {
    boys.remove(boy);
}

@Override
public void notifyObservers() {
    
    for(int i=0;i<boys.size();i++){
        boys.get(i).update();
    }
}

public void hungrey(){
    notifyObservers();
}

}

测试类ObserverTest

public class ObserverTest {

public static void main(String[] args) {
    
    //小天使来到世界上
    Girl girl = new Girl();
    //护花使者A
    BoyA boyA = new BoyA(girl);
    //护花使者B
    BoyB boyB = new BoyB(girl);
    //护花使者C
    BoyC boyC = new BoyC(girl);
    //出大事了,小可爱饿了
    girl.hungrey();
}
}

结果:

result

好了,到此我们观察者模式就简单地描述完了。回到开篇前我们的两个问题,我们现在这种模式有没有解决问题呢?

  1. 作为女孩,首先我要知道谁喜欢我啊,不然通知错人了怎么办?(实际情况是,尽管你知道有人喜欢你,但是这个人是谁你知道吗?)

我们现在在Girl(Subject实现类)中维护一个数组或者队列,用来保存观察者。并不需要知道观察者是谁。我们在创建Girl的时候将队列初始化(换句话说,我知道肯定有人喜欢我,但是具体是谁,我并不需要知道)。

  1. 又有人喜欢女孩了,也想照顾她。但是不想让女孩知道,只想默默地付出。按照我们现在这种写法,只有女孩知道谁喜欢她才会通知到,不知道的就不通知了。

现在即便是有BoyC,BoyD出现,在创建的时候就自己偷偷关注了女孩,当女孩饿了的时候,并不需要改变原有的代码就可实现通知所有的。(多好,我们的这种实现方式。毕竟暗恋也是美好的)。

然而在实际中,主题Subject在通知观察者Observer的时候会携带一些数据,这就不得不提一下观察者模式的两种模式:推(push)模式和拉(pull)模式

推模型:顾名思义,就是我不管你Observer想要什么,我给你什么你接收什么就是了。这种模式的使用场景就是需求较简单,且Subject和Observer类协商好返回数据的类型。弊端当然也很明显就是众Observer口难调,我不知道你想要什么,大家返回都一样。可以想象成任性的女孩,我给你什么你要也得接受,不要也得接受。

全写的话量有点大,所以只贴部分代码:

public interface Subject {
public void addObserver(Observer boy);
public void removeObserver(Observer boy);
public void notifyObservers(int state);
}

可以看出,这个地方我们将state作为参数传出去,然后在Observer中被动接收。

public interface Observer {

public void update(int state);
//举个例子,这时候我们就可以在BoyA,BoyB...等中针对具体状态作出相应动作。比如饿了怎么样,困了怎么样......
}

拉模型:就是观察者(Observer)自己站起来了,想要什么就给什么。一般我们把主题类该类对象作为参数传递出去,然后观察者可以利用反射等方式拿到自己想要的,推模型的问题解决了,大家(众多观察者)可以各取所需。可以这么理解,付出这么多,女孩追到手了,把自己都交给你了。

public interface Subject {
public void addObserver(Observer boy);
public void removeObserver(Observer boy);
public void notifyObservers(Subject subject);
}

Girl

...

@Override
public void notifyObservers(Subject subject) {
    
    for(int i=0;i<boys.size();i++){
        boys.get(i).update(subject);
    }
}

public void hungrey(){
    
    notifyObservers(this);//将自己传出去
}
...

Observer

public interface Observer {

public void update(Subject subject);//人都得到了,想知道什么还难吗?

}

好了,观察者模式到此介绍完毕,大家有什么问题可以与我讨论。当然文中错误在所难免,恳请批评指正。

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

推荐阅读更多精彩内容