Spring 引用的7种设计模式

作者: 黄靠谱 | 来源:发表于2019-02-17 12:10 被阅读26次

概述

  • 工厂模式解决的问题:解耦、创建时干预、统一管理
  • 单例模式解决的问题:可以共享的资源就不要重复创建,特别是创建起来成本很高的资源,比如数据源
  • 代理模式解决的问题:既不修改基类(无侵入),又可以灵活的扩展它的功能,而且这种扩展是可以复用的,比如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里面配置的 拦截器,按照业务的需求来按照一定的顺序自由组合起来,实现特定的业务场景

相关文章

网友评论

    本文标题:Spring 引用的7种设计模式

    本文链接:https://www.haomeiwen.com/subject/twkgeqtx.html