设计模式回顾(三)

三、行为型模式

行为型模式是使用较为频繁也是相对有些理解难度的模式。主要针对不同的业务需求对对象的行为进行优化和改进,使之更加合理,让业务更加流畅稳定,同时也更便于代码阅读理解。
行为型模式主要包括观察者模式、责任链模式、命令模式、迭代器模式、中介者模式、备忘录模式、状态模式、策略模式、访问者模式。以下详细介绍。

观察者模式

这种模式主要用于监听对象的行为或状态变化,当被观察对象的状态发生改变时,迅速通知到观察者,因此通常是一对多的关系,即一个被观察者多个观察者。比如警察常常要接听报警电话,观察者就是警察,观察对象就是每一个报警求助的人,观察途径就是电话。另外求助者可能向警察求助,也有可能想亲朋好友求助,所以可能会有多个观察者。如下图:


image.png

报警电话为一个接口,警察和求助者都持有,这是他们互通信息的一个通道。

public interface DangerListener {
    void somethingHappen(String event);
}

求助者拥有一个电话本,只要有人告诉他号码他就会存下来,和他断交就会删掉这个号码,因此他有一个电话集合。当他遇到危险,就会把这个信息向电话集合中的人群发扩散以获得帮助。

public class HelperSeeker {

    private List<DangerListener> listeners = new ArrayList<>();
    private static HelperSeeker seeker;

    public static HelperSeeker getInstance(){
        if (seeker == null){
            seeker = new HelperSeeker();
        }
        return seeker;
    }

    public void registerDanger(DangerListener listener){
        listeners.add(listener);
    }

    public void unRegisterDanger(DangerListener listener){
        listeners.remove(listener);
    }

    public void help(){
        for (DangerListener listener:listeners){
            listener.somethingHappen("help");
        }
    }

}

警察拥有这个求助者的信息,并告诉他自己的联系方式,当收到他的求助信息时就会采取保护行动。当警察换岗位或者离职或是退休之后就会告诉该求助者不用保存自己的号码了。

public class ObserverPolice {

    private DangerListener listener = new DangerListener() {
        @Override
        public void somethingHappen(String event) {
            System.out.println("ObserverPolice receive msg for "+event);
        }
    };

    public void action(){
        HelperSeeker seeker = HelperSeeker.getInstance();
        seeker.registerDanger(listener);
    }


    public void onChange(){
        HelperSeeker.getInstance().unRegisterDanger(listener);
    }

}

责任链模式

责任链模式即接受者在处理请求的时候创建了一个对象链来进行事件处理,这个对象链中每个接受者都拥有下一个的引用,当自己能处理该请求时则自己处理掉,不能处理的话往下传递,直至处理完毕。类似“击鼓传花”。
例如,当前要实现一个打印日志的功能,日志分三个级别,一般日志,错误日志和文件日志。等级越高的日志越是要谨慎打印,等级低的限制宽松一些,一般的都要打印。
还举个例子,屏幕上有一个弹窗,弹窗上会进行一些操作也会弹窗,弹窗上又有可能弹窗,简而言之有多重弹窗,假如现在有三层弹窗,在最底层的页面突然收到一个消息,要求把所有弹窗都消失掉。这时应该怎么处理呢?
由于最底层的页面是不应该全部持有这三个弹窗的引用的,否则就会造成管理混乱,所以在收到通知的时候只需要在onDimiss()方法中将上面的一个弹窗取消掉就行,而上一个弹窗在取消的时候只需要判断它上面有没有弹窗,有的话就取消就行了,如此类推。当一个弹窗取消的时候,它上面的所有弹窗都会取消掉。每一个弹窗都持有下一个弹窗的引用,都只负责自己是否取消。


image.png

代码如下:
基类logger

public abstract class Logger {

    public static final int LEVEL_CONSOLE = 0;
    public static final int LEVEL_ERROR = 1;
    public static final int LEVEL_FILE = 2;

    private Logger nextLogger = getNextLogger();

    protected abstract Logger getNextLogger();

    private int curLevel = getLevel();

    protected abstract int getLevel();

    protected void loggerMsg(int level,String msg){
        if (curLevel <= level){
            logger(msg);
            if (nextLogger != null){
                nextLogger.logger(msg);
            }
        }
    }

    protected abstract void logger(String msg);

}

三个子类ConsoleLogger,ErrorLogger和FileLogger

public class ConsoleLogger extends Logger {

    @Override
    protected Logger getNextLogger() {
        return new ErrorLogger();
    }

    @Override
    protected int getLevel() {
        return Logger.LEVEL_CONSOLE;
    }

    @Override
    protected void logger(String msg) {
        System.out.println("ConsoleLogger " + msg);
    }
}
public class ErrorLogger extends Logger{
    @Override
    protected Logger getNextLogger() {
        return new FileLogger();
    }

    @Override
    protected int getLevel() {
        return Logger.LEVEL_ERROR;
    }

    @Override
    protected void logger(String msg) {
        System.out.println("ErrorLogger "+msg);
    }
}
public class FileLogger extends Logger{
    @Override
    protected Logger getNextLogger() {
        return null;
    }

    @Override
    protected int getLevel() {
        return Logger.LEVEL_FILE;
    }

    @Override
    protected void logger(String msg) {
        System.out.println("FileLogger "+msg);
    }
}

测试

public class Test {

    public static void main(String[] args) {
        ConsoleLogger logger = new ConsoleLogger();
        logger.loggerMsg(Logger.LEVEL_ERROR,"error msg");
    }

}

打印结果

ConsoleLogger error msg
ErrorLogger error msg

命令模式

命令模式就是将命令者和命令本身以及命令对象分别进行封装,这样再多的命令都可以简化。达到了对象之间的解耦。
将命令者和命令都抽象成接口,命令者持有命令引用而不关心命令如何去实现。一个典型的例子就是Android当中的各个view。每个view都拥有自己的一套绘制流程,onMeasure(),onLayout(),onDraw(),等,view作为一个基类抽象了这些方法,而同时每个view又都是可以被点击的,即view的触摸事件分发,这一套也抽象成了接口方法。基类持有这一引用,具体的实现则由各自去实现。例如onClick(),onLongClick(),onDrag()等。如下图:


image.png

View

public abstract class View {
    protected TouchEvent touchEvent;
    protected String name;

    public View(String name,TouchEvent touchEvent) {
        this.name = name;
        this.touchEvent = touchEvent;
    }

    public void touch(){
        touchEvent.onTouch();
    }

    public String getName(){
        return name;
    }

}

TextView

public class TextView extends View{
    public TextView(String name, TouchEvent touchEvent) {
        super(name, touchEvent);
    }
}

TouchEvent

public interface TouchEvent {
    void onTouch();
}

DragEvent

public class DragEvent implements TouchEvent{
    @Override
    public void onTouch() {
        System.out.println("DragEvent ...");
    }
}

ClickEvent

public class ClickEvent implements TouchEvent{
    @Override
    public void onTouch() {
        System.out.println("ClickEvent ...");
    }
}

测试代码:

    public static void main(String[] args) {
        ClickEvent clickEvent = new ClickEvent();
        DragEvent dragEvent = new DragEvent();
        TextView textView1 = new TextView("textView1",clickEvent);
        System.out.print(textView1.getName()+" ");
        textView1.touch();
        TextView textView2 = new TextView("textView2",dragEvent);
        System.out.print(textView2.getName()+" ");
        textView2.touch();
    }

迭代器模式

这种模式用于顺序访问集合对象的元素,不需要知道集合对象的底层表示。一般用于集合中,例如iterator接口,它的实现类有很多种如ArrayList(),LinkList()等等,具体可以直接看jdk源码。

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

推荐阅读更多精彩内容

  • 设计模式(Design pattern)是一套被反复使用、多数人知晓的、经过分类编目的、代码设计经验的总结。使用设...
    闽越布衣阅读 486评论 0 0
  • 23 个设计模式 课程:https://b23.tv/Mxr9b6 课件:https://github.com/r...
    YueJZ阅读 395评论 0 0
  • 本篇是四部曲的第三篇,第一篇请点这里iOS设计模式四部曲(一):创建型模式,第二篇请点击这里iOS设计模式四部曲(...
    SuperMario_Nil阅读 2,822评论 4 23
  • 行为型模式用于描述程序在运行时复杂的流程控制,即描述多个类或对象之间怎样相互协作共同完成单个对象都无法单独完成的任...
    放下梧菲阅读 277评论 0 1
  • 我是黑夜里大雨纷飞的人啊 1 “又到一年六月,有人笑有人哭,有人欢乐有人忧愁,有人惊喜有人失落,有的觉得收获满满有...
    陌忘宇阅读 8,523评论 28 53