设计模式解析四 模板方法模式和外观模式

一. 前言

讲到第四篇,其实已经把最常用的设计模式讲了一半了,今天讲策略模式的表兄弟,模板方法模式。

二. 模板方法模式

模板方式模式会被称为策略模式的兄弟的原因,是大家都封装算法,但策略模式注重的是封装一整个算法族,他的抽象类只有一个接口。但模板方法模式不同,他封装的是算法的一部分,抽象类里定义好了整个算法的模板,而其中的一部分算法则交由子类实现。
还是来举个蒸包子的例子,蒸包子我们来拆分一下,看看有哪些步骤:

  • 和面
  • 准备包子馅
  • 包包子

我们大致将蒸包子的步骤分为上面四个,有哪些步骤是一样的呢?可以任务和面的步骤都是一样的,蒸的步骤是一样的,但是准备包子馅不一样,有猪肉的,牛肉的,梅干菜的,豆沙的。包的动作也不太一样,有的普通大包子,有的是小笼包。
那么我们把动作一样的部分写死,把可变的部分提供为抽象来供大家实现,是不是就能蒸出好吃的包子了呢?
接下来写代码:

public abstract class AbstractSteamedBun {
    public final void steameBun() {
        kneadDough();
        prepareStuffing();
        wrapBun();
        steam();
    }
    protected abstract void wrapBun();
    protected abstract void prepareStuffing();
    private void kneadDough() {
        System.out.println("和面...");
    }
    private void steam() {
        System.out.println("开始蒸包子");
    }
}

姐下面我们就来蒸个牛肉包:

public class SteamedBeefBun extends AbstractSteamedBun {
    @Override
    protected void wrapBun() {
        System.out.println("开始包牛肉包子");
    }
    @Override
    protected void prepareStuffing() {
        System.out.println("开始准备牛肉包子馅");
    }
}

运行:

    public static void main(String[] args) {
        AbstractSteamedBun bun = new SteamedBeefBun();
        bun.steameBun();
    }
和面...
开始准备牛肉包子馅
开始包牛肉包子
开始蒸包子

这就是模板方法模式,模板方法模式的应用也很广,很多地方随处可见,
比如最常见的排序Collections.sort(list),我们经常使用这个排序工具类,我们应该从没想过他的排序是用的什么算法,他是怎么排的,我们只知道我们需要让我们的实体实现Comparable接口,实际上他的内部把整个排序的逻辑步骤都为我们固定好了,我们只需要实现值的比较即可,只是这里把这个模板方法设计成了接口跟算法进行了分离。
再有Spring框架的容器加载过程也有一个模板方法设计模式,Spring的抽象容器类AbstractApplicationContext的源码:

@Override
public void refresh() throws BeansException, IllegalStateException {
    synchronized (this.startupShutdownMonitor) {
        // Prepare this context for refreshing.
        prepareRefresh();
        // Tell the subclass to refresh the internal bean factory.
        ConfigurableListableBeanFactory beanFactory = obtainFreshBeanFactory();
        // Prepare the bean factory for use in this context.
        prepareBeanFactory(beanFactory);
        try {
            // Allows post-processing of the bean factory in context subclasses.
            postProcessBeanFactory(beanFactory);
            // Invoke factory processors registered as beans in the context.
            invokeBeanFactoryPostProcessors(beanFactory);
            // Register bean processors that intercept bean creation.
            registerBeanPostProcessors(beanFactory);
            // Initialize message source for this context.
            initMessageSource();
            // Initialize event multicaster for this context.
            initApplicationEventMulticaster();
            // Initialize other special beans in specific context subclasses.
            onRefresh();
            // Check for listener beans and register them.
            registerListeners();
            // Instantiate all remaining (non-lazy-init) singletons.
            finishBeanFactoryInitialization(beanFactory);
            // Last step: publish corresponding event.
            finishRefresh();
        }
        catch (BeansException ex) {
            if (logger.isWarnEnabled()) {
                logger.warn("Exception encountered during context initialization - " +
                        "cancelling refresh attempt: " + ex);
            }
            // Destroy already created singletons to avoid dangling resources.
            destroyBeans();
            // Reset 'active' flag.
            cancelRefresh(ex);
            // Propagate exception to caller.
            throw ex;
        }
        finally {
            // Reset common introspection caches in Spring's core, since we
            // might not ever need metadata for singleton beans anymore...
            resetCommonCaches();
        }
    }
}

其中的obtainFreshBeanFactory()方法中有调用一个refreshBeanFactory()方法,该方法是一个抽象方法。
然后postProcessBeanFactory(beanFactory)也是一个空实现的方法,
onRefresh()方法也是一个空实现的方法,这些方法都是等待子类来实现的方法,尽管是非必要的。

定义

模板方法模式在一个方法中定义一个算法的骨架,而将一些步骤延迟到子类中。模板方法使得子类可以在不改变算法架构的情况下,重新定义算法中的某些步骤。

三. 外观模式

外观模式和模板方法模式没什么关系,写在这只是因为觉得一章就写一个设计模式有点太少了.
其实这个模式跟适配器模式反而有一点相似,适配器模式通过把一个接口适配成另一个接口,目的是为了让不兼容的两个系统的接口能够运行。而外观模式也改变接口,但是他改变的接口的原因是为了简化接口,之所以这么称呼,是因为他将一个或数个类的复杂的一切都隐藏在背后,只露出一个干净美好的外观。
又到了有趣的举例子环节,我是个小米智能家居的拥捧者,家里也买了很多小米的智能家居,我们家客厅有一个孔灯,亮度较暗,还有一个大灯,非常亮,平时就开孔灯,偶尔就打开大灯然后关闭孔灯,都是可以通过小爱同学来进行控制的,所以我的日常是这样:

  • 当我的孔灯亮着想开大灯的时候我会这样
    小爱同学,打开大灯
    小爱同学,关闭孔灯
  • 当我的大灯亮着我想开孔灯的时候我会这样
    小爱同学,打开孔灯
    小爱同学,关闭大灯

但是用着用着我就觉得好麻烦啊,总是我的目标就是开一个灯,关另一个灯,能不能就喊一条指令啊,有一天突然发现小米有一个可以设置组合指令的地方:


智能

配置了这个,然后在训练小爱同学收到开孔灯指令的时候就执行这个指令,以后再也不用每次都要喊两个指令那么麻烦了。
那么这个组合指令对于我们的小爱同学或者说对于我们来说就是一个外观模式,我们调用者不关心你内部复杂的细节,只关心我调用后要达到的效果,至于内部你是执行了一条指令,还是执行了两条指令对于调用者来说并不重要,引用百度百科对外观模式的评价:

  • 实现了子系统与客户端之间的松耦合关系。
  • 客户端屏蔽了子系统组件,减少了客户端所需处理的对象数目,并使得子系统使用起来更加容易。

另外再举个例子,我们经常使用日志系统,日志系统有一个组件叫做 slf4j,大名鼎鼎了,那么大家有没有想过为什么会取这样的一个名字?
其实他的全名是:Simple Logging Facade for Java。为java提供的简单日志外观,从名字就告诉了我们,这是一个外观模式,实际上slf4j本身并没有打印日志的能力,他只是定义了一套日志打印的接口Logger接口还有ILoggerFactory接口等,真正打印日志的还是 log4jlogback
我们来看一下日志系统家族:

日志系统

slf4j和common-logging都是日志接口,而后面的才是实现,那么为什么叫外观模式呢
org.slf4j.Logger logger = org.slf4j.LoggerFactory.getLogger(Application.class);
这句代码是我们通常获取日志打印对象的代码,使用的都是slf4j包中的类,在LoggerFactory.getLogger中,它会为我们取寻找logger的实现和装载过程,从而我们对我们隐藏内部的日志打印细节,所以我们才能这么方便的使用日志打印系统,讲到这里就再深入的去看一下里面的源码:

    public static Logger getLogger(Class<?> clazz) {
        Logger logger = getLogger(clazz.getName());
        if (DETECT_LOGGER_NAME_MISMATCH) {
            Class<?> autoComputedCallingClass = Util.getCallingClass();
            if (autoComputedCallingClass != null && nonMatchingClasses(clazz, autoComputedCallingClass)) {
                Util.report(String.format("Detected logger name mismatch. Given name: \"%s\"; computed name: \"%s\".", logger.getName(),
                                autoComputedCallingClass.getName()));
                Util.report("See " + LOGGER_NAME_MISMATCH_URL + " for an explanation");
            }
        }
        return logger;
    }
    public static Logger getLogger(String name) {
        ILoggerFactory iLoggerFactory = getILoggerFactory();
        return iLoggerFactory.getLogger(name);
    }

重点在getILoggerFactory方法中,如果我们去翻找logback或者log4j就能找到,它里面都有一个工厂类是实现了ILoggerFactory的,而且他们的日志打印类也是都是实现了Logger类的,你要说为什么他们会遵照这个标准,那是因为slf4j和logback和log4j都是一个人做的,膜拜大神。

    public static ILoggerFactory getILoggerFactory() {
        if (INITIALIZATION_STATE == UNINITIALIZED) {
            synchronized (LoggerFactory.class) {
                if (INITIALIZATION_STATE == UNINITIALIZED) {
                    INITIALIZATION_STATE = ONGOING_INITIALIZATION;
                    performInitialization();
                }
            }
        }
        switch (INITIALIZATION_STATE) {
        case SUCCESSFUL_INITIALIZATION:
            return StaticLoggerBinder.getSingleton().getLoggerFactory();
        case NOP_FALLBACK_INITIALIZATION:
            return NOP_FALLBACK_FACTORY;
        case FAILED_INITIALIZATION:
            throw new IllegalStateException(UNSUCCESSFUL_INIT_MSG);
        case ONGOING_INITIALIZATION:
            // support re-entrant behavior.
            // See also http://jira.qos.ch/browse/SLF4J-97
            return SUBST_FACTORY;
        }
        throw new IllegalStateException("Unreachable code");
    }
}

在这里,会判断是否已经进行过日志容器初始化,如果没有初始化则会调用
performInitialization方法进行初始化,在这个方法里有一个bind方法是重点。

    private final static void bind() {
        try {
            Set<URL> staticLoggerBinderPathSet = null;
            // skip check under android, see also
            // http://jira.qos.ch/browse/SLF4J-328
            if (!isAndroid()) {
                staticLoggerBinderPathSet = findPossibleStaticLoggerBinderPathSet();
                reportMultipleBindingAmbiguity(staticLoggerBinderPathSet);
            }
            // the next line does the binding
            StaticLoggerBinder.getSingleton();
            INITIALIZATION_STATE = SUCCESSFUL_INITIALIZATION;
            reportActualBinding(staticLoggerBinderPathSet);
            fixSubstituteLoggers();
            replayEvents();
            // release all resources in SUBST_FACTORY
            SUBST_FACTORY.clear();
        } catch (NoClassDefFoundError ncde) {
            String msg = ncde.getMessage();
            if (messageContainsOrgSlf4jImplStaticLoggerBinder(msg)) {
                INITIALIZATION_STATE = NOP_FALLBACK_INITIALIZATION;
                Util.report("Failed to load class \"org.slf4j.impl.StaticLoggerBinder\".");
                Util.report("Defaulting to no-operation (NOP) logger implementation");
                Util.report("See " + NO_STATICLOGGERBINDER_URL + " for further details.");
            } else {
                failedBinding(ncde);
                throw ncde;
            }
        } catch (java.lang.NoSuchMethodError nsme) {
            String msg = nsme.getMessage();
            if (msg != null && msg.contains("org.slf4j.impl.StaticLoggerBinder.getSingleton()")) {
                INITIALIZATION_STATE = FAILED_INITIALIZATION;
                Util.report("slf4j-api 1.6.x (or later) is incompatible with this binding.");
                Util.report("Your binding is version 1.5.5 or earlier.");
                Util.report("Upgrade your binding to version 1.6.x.");
            }
            throw nsme;
        } catch (Exception e) {
            failedBinding(e);
            throw new IllegalStateException("Unexpected initialization failure", e);
        }
    }

这里有一个findPossibleStaticLoggerBinderPathSet方法,会去利用类加载器加载一个实现类

    private static String STATIC_LOGGER_BINDER_PATH = "org/slf4j/impl/StaticLoggerBinder.class";

    static Set<URL> findPossibleStaticLoggerBinderPathSet() {
        // use Set instead of list in order to deal with bug #138
        // LinkedHashSet appropriate here because it preserves insertion order
        // during iteration
        Set<URL> staticLoggerBinderPathSet = new LinkedHashSet<URL>();
        try {
            ClassLoader loggerFactoryClassLoader = LoggerFactory.class.getClassLoader();
            Enumeration<URL> paths;
            if (loggerFactoryClassLoader == null) {
                paths = ClassLoader.getSystemResources(STATIC_LOGGER_BINDER_PATH);
            } else {
                paths = loggerFactoryClassLoader.getResources(STATIC_LOGGER_BINDER_PATH);
            }
            while (paths.hasMoreElements()) {
                URL path = paths.nextElement();
                staticLoggerBinderPathSet.add(path);
            }
        } catch (IOException ioe) {
            Util.report("Error getting resources from path", ioe);
        }
        return staticLoggerBinderPathSet;
    }

而这个实现类就是StaticLoggerBinder.class,去看log4j和logback就会发现,这两个包都有这个类,在这个类在实例化的时候会创建一个ILoggerFactory的实例,拿到这个工厂就可以创建各自的日志打印对象了。
你看,这个内部有很多复杂的东西,但是外观模式帮我们搞定以后,我们只需要调用一句话:
org.slf4j.Logger logger = org.slf4j.LoggerFactory.getLogger(Application.class);
这就是外观模式的魅力了。

定义

外观模式提供了一个统一的接口,用来访问子系统中的一群接口,外观定义了一个高层接口,让子系统更容易使用。

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