Spring 引用的7种设计模式

概述

  • 工厂模式解决的问题:解耦、创建时干预、统一管理
  • 单例模式解决的问题:可以共享的资源就不要重复创建,特别是创建起来成本很高的资源,比如数据源
  • 代理模式解决的问题:既不修改基类(无侵入),又可以灵活的扩展它的功能,而且这种扩展是可以复用的,比如AspectJ、CGLIB、JDK动态代理
  • 观察者模式解决的问题:事件通知,比如zk节点的watch机制,比如tomcat的启动机制(创建IOC、MVC容器)
  • 模板模式解决的问题:代码冗余,通过模板类+业务类作为参数解决。比如JDBCTemplate,模板方法处理 创建连接、处理异常、释放资源等操作,业务类执行自己的sql
  • 策略模式解决的问题:同一类型业务的类,有很多公用的流程和方法,只是在核心方法上略有区别,为了降低代码的冗余度,单独把不同的方法抽象成一个接口,各自业务类实现自己的核心方法
  • 责任链模式解决的问题:把一套流程拆分成不同的Handler,使用的时候根据业务场景拼装,可以非常灵活和低耦合的实现特定的业务流程。

工厂方法模式,

  • 工厂模式的优点:解耦、统一调度和管理。
  • 工厂模式的角色:工厂类、产品接口类、产品类s

简单工厂模式:通过beanID+Bean全路径生产bean(反射),具体由BeanFactory的实现类来生产。还提供默认单例缓存功能。
比如容器级别的生命周期干预、父子容器、单例模式的控制,这些都是由工厂端(IOC)来做的,如果通过new的形式创建Bean,实现起来就会更复杂。
IOC是在生产Bean的时候,check需不需要干预,check是否单例,是的话,丢到singotonObject里面去,需不需要动态代理。

        BeanFactory bf = new XmlBeanFactory(new ClassPathResource("appcxt-context.xml"));
        Car car=bf.getBean("car", Car.class);

单例模式

核心方法通过AbstractBeanFactory实现的

  1. 最开始会做一个Eagerly check,如果是单例且已经缓存,后面就不用管了,直接返回单例
  2. 做完一些校验之后,再加锁(this.singletonObjects 这个对象),Double Check之后,再创建Bean,创建完之后缓存在singletonObjects里面
  3. 创建Bean的时候加了锁,addSingleton时候也重入了该锁
public Object getSingotonByBeanName(String beanName){
    Object bean=getSingleton(String beanName, boolean allowEarlyReference) ;
    if(bean==null){
        synchronized (this.singletonObjects) {
                //双重校验
                Object singletonObject = this.singletonObjects.get(beanName);
                if (singletonObject == null) {
                    singletonObject = singletonFactory.createBean();
                    addSingleton(beanName, singletonObject);
                }
                return singletonObject;
            }
    }
}

代理模式

JDK动态代理、CGLIB代理模式、AspectJ

  • AspectJ 编译期代理
    -JDK动态代理,CGLIB 代理,都是动态编译

  • JDK动态代理:反射原理基于接口的方法做代理,代理类只包含接口里面的所有方法。

  • CGLIB: 是Encacher工具类,继承代理类,植入代理逻辑.

模板模式

解决的问题:减少冗余代码,通过模板方法+callback对象
实现方法:模板类提供模板方法,调用类接口提供自定义的通用方法,调用类作为参数,在执行完模板方法时,出发自己的业务逻辑代码
比如JDBCTemplate:封装了 获得数据库连接,处理事务,处理异常,关闭资源等等通用方法,执行完之后,调用callback的不一样的业务逻辑

public <T> T execute(StatementCallback<T> action)  {
        Connection con = DataSourceUtils.getConnection(getDataSource());
        Statement stmt = null;
        try {
            Connection conToUse = con;
            stmt = conToUse.createStatement();
            applyStatementSettings(stmt);
            Statement stmtToUse = stmt;
            T result = action.doInStatement(stmtToUse);
            return result;
        }
        catch (SQLException ex) {
            JdbcUtils.closeStatement(stmt);
            stmt = null;
            DataSourceUtils.releaseConnection(con, getDataSource());
            con = null;
            throw getExceptionTranslator().translate("StatementCallback", getSql(action), ex);
        }
        finally {
            JdbcUtils.closeStatement(stmt);
            DataSourceUtils.releaseConnection(con, getDataSource());
        }
    }

观察者模式

  • 观察者模式解决的问题:事件通知,比如zk节点的watch机制,比如tomcat的启动机制(创建IOC、MVC容器)
  • 观察者模式的角色:被观察者,观察者接口(定义通知的回调方法)、具体的观察者(可能是多个)
  • 原理:被观察者开放接口接受注册,当被观察者根据业务设计,在一定的时机下,触发已经注册了的观察者的通知方法
    比如web容器,ContextLoaderListener注册,容器启动的时候,会根据指定的Spring配置文件,创建IOC容器。

策略模式

  1. Spring MVC中引用了策略模式解决了不同类型的Request,可以共用DispatchServlet.doDispach()方法。

  2. 程序会根据HandlerMapping中反馈的Handler的类型来选择对于的适配器接口的实现类,最终实现不同的处理逻辑

  3. HandlerAdapter是一个接口,所有的扩展功能类都实现这个接口的 handler()方法,这样做的好处是节省内存,用到那种类型的Handler就加载哪个Handler,而不是一股脑的全加载进来
    DispatcherServlet.doDispatch

protected void doDispatch(HttpServletRequest request, HttpServletResponse response) throws Exception {
    ModelAndView mv = null;

    // Determine handler for the current request.
    mappedHandler = getHandler(processedRequest);

    // Determine handler adapter for the current request.
    HandlerAdapter ha = getHandlerAdapter(mappedHandler.getHandler());

    // Actually invoke the handler.
    mv = ha.handle(processedRequest, response, mappedHandler.getHandler());

    applyDefaultViewName(processedRequest, mv);
    mappedHandler.applyPostHandle(processedRequest, response, mv);
    processDispatchResult(processedRequest, response, mappedHandler, mv, dispatchException);
    
}
  1. HandlerAdapter 代表的不同类型的Handler们


    image

责任链模式

灵活性在于:

  1. 可以根据业务规则配置不同顺序的拦截器(责任链)
  2. 可以根据业务规则配置不同种类的拦截器(责任链)

Spring里面配置的 拦截器,按照业务的需求来按照一定的顺序自由组合起来,实现特定的业务场景

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

推荐阅读更多精彩内容