Spring框架是每个java程序猿入门级的框架也是最重要的框架,而Spring中也采用了很多的设计模式,这些也会成为我们面试过程中经常会问到的问题,所以本文就整理出Spring中具体使用的哪些设计模式。


Java单例模式

Java原型模式(prototype)

Java模板模式(template)

Java观察者模式(Observer)

Java工厂模式

Java适配器模式(adapter)

Java装饰者模式(decorator)

Java代理模式

Java策略模式(Strategy)


@

Spring使用的设计模式

1.单例模式

  单例模式应该是大家印象最深的一种设计模式了。在Spring中最明显的使用场景是在配置文件中配置注册bean对象的时候设置scope的值为singleton

  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <beans xmlns="http://www.springframework.org/schema/beans"
  3. xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
  4. xsi:schemaLocation="http://www.springframework.org/schema/beans
  5. http://www.springframework.org/schema/beans/spring-beans.xsd">
  6. <bean class="com.dpb.pojo.User" id="user" scope="singleton">
  7. <property name="name" value="波波烤鸭"></property>
  8. </bean>
  9. </beans>

源码实现:AbstractBeanFactory的getBean方法中

  1. protected Object getSingleton(String beanName, boolean allowEarlyReference) {
  2. Object singletonObject = this.singletonObjects.get(beanName);
  3. if (singletonObject == null && isSingletonCurrentlyInCreation(beanName)) {
  4. synchronized (this.singletonObjects) {
  5. singletonObject = this.earlySingletonObjects.get(beanName);
  6. if (singletonObject == null && allowEarlyReference) {
  7. ObjectFactory<?> singletonFactory = this.singletonFactories.get(beanName);
  8. if (singletonFactory != null) {
  9. singletonObject = singletonFactory.getObject();
  10. this.earlySingletonObjects.put(beanName, singletonObject);
  11. this.singletonFactories.remove(beanName);
  12. }
  13. }
  14. }
  15. }
  16. return (singletonObject != NULL_OBJECT ? singletonObject : null);
  17. }

双重判断加锁的实现!!!

2.原型模式

  原型模式也叫克隆模式,Spring中该模式使用的很明显,和单例一样在bean标签中设置scope的属性为prototype即表示该bean以克隆的方式生成

  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <beans xmlns="http://www.springframework.org/schema/beans"
  3. xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
  4. xsi:schemaLocation="http://www.springframework.org/schema/beans
  5. http://www.springframework.org/schema/beans/spring-beans.xsd">
  6. <bean class="com.dpb.pojo.User" id="user" scope="prototype">
  7. <property name="name" value="波波烤鸭"></property>
  8. </bean>
  9. </beans>

3.模板模式

  模板模式的核心是父类定义好流程,然后将流程中需要子类实现的方法就抽象话留给子类实现,Spring中的JdbcTemplate就是这样的实现。我们知道jdbc的步骤是固定的(

  1. 加载驱动,
  2. 获取连接通道,
  3. 构建sql语句.
  4. 执行sql语句,
  5. 关闭资源),

  在这些步骤中第3步和第四步是不确定的,所以就留给客户实现,而我们实际使用JdbcTemplate的时候也确实是只需要构建SQL就可以了.这就是典型的模板模式。我们以query方法为例来看下JdbcTemplate中的代码

  1. // 在execute方法中定义好了jdbc操作的流程
  2. // action.doInStatement(stmtToUse);是回调方法也就是钩子
  3. @Override
  4. public <T> T execute(StatementCallback<T> action) throws DataAccessException {
  5. Assert.notNull(action, "Callback object must not be null");
  6. Connection con = DataSourceUtils.getConnection(getDataSource());
  7. Statement stmt = null;
  8. try {
  9. Connection conToUse = con;
  10. if (this.nativeJdbcExtractor != null &&
  11. this.nativeJdbcExtractor.isNativeConnectionNecessaryForNativeStatements()) {
  12. conToUse = this.nativeJdbcExtractor.getNativeConnection(con);
  13. }
  14. stmt = conToUse.createStatement();
  15. applyStatementSettings(stmt);
  16. Statement stmtToUse = stmt;
  17. if (this.nativeJdbcExtractor != null) {
  18. stmtToUse = this.nativeJdbcExtractor.getNativeStatement(stmt);
  19. }
  20. T result = action.doInStatement(stmtToUse);
  21. handleWarnings(stmt);
  22. return result;
  23. }
  24. catch (SQLException ex) {
  25. // Release Connection early, to avoid potential connection pool deadlock
  26. // in the case when the exception translator hasn't been initialized yet.
  27. JdbcUtils.closeStatement(stmt);
  28. stmt = null;
  29. DataSourceUtils.releaseConnection(con, getDataSource());
  30. con = null;
  31. throw getExceptionTranslator().translate("StatementCallback", getSql(action), ex);
  32. }
  33. finally {
  34. JdbcUtils.closeStatement(stmt);
  35. DataSourceUtils.releaseConnection(con, getDataSource());
  36. }
  37. }

query方法

  1. @Override
  2. public <T> T query(final String sql, final ResultSetExtractor<T> rse) throws DataAccessException {
  3. Assert.notNull(sql, "SQL must not be null");
  4. Assert.notNull(rse, "ResultSetExtractor must not be null");
  5. if (logger.isDebugEnabled()) {
  6. logger.debug("Executing SQL query [" + sql + "]");
  7. }
  8. // 实现模板中预留的功能
  9. class QueryStatementCallback implements StatementCallback<T>, SqlProvider {
  10. @Override
  11. public T doInStatement(Statement stmt) throws SQLException {
  12. ResultSet rs = null;
  13. try {
  14. // 此处具体执行查询操作
  15. rs = stmt.executeQuery(sql);
  16. ResultSet rsToUse = rs;
  17. if (nativeJdbcExtractor != null) {
  18. rsToUse = nativeJdbcExtractor.getNativeResultSet(rs);
  19. }
  20. // 处理数据封装操作
  21. return rse.extractData(rsToUse);
  22. }
  23. finally {
  24. JdbcUtils.closeResultSet(rs);
  25. }
  26. }
  27. @Override
  28. public String getSql() {
  29. return sql;
  30. }
  31. }
  32. return execute(new QueryStatementCallback());
  33. }

4.观察者模式

  观察者模式定义的是对象间的一种一对多的依赖关系,当一个对象的状态发生改变时,所有依赖于它的对象都得到通知并被自动更新。使用比较场景是在监听器中而spring中Observer模式常用的地方也是listener的实现。如ApplicationListener。 Spring中的事件监听请参考我的另一篇文章

Spring之事件监听(观察者模型)

5.工厂模式

简单工厂模式

  简单工厂模式就是通过工厂根据传递进来的参数决定产生哪个对象。Spring中我们通过getBean方法获取对象的时候根据id或者name获取就是简单工厂模式了。

  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <beans xmlns="http://www.springframework.org/schema/beans"
  3. xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
  4. xmlns:context="http://www.springframework.org/schema/context"
  5. xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd
  6. http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-4.3.xsd">
  7. <context:annotation-config/>
  8. <bean class="com.dpb.pojo.User" id="user" >
  9. <property name="name" value="波波烤鸭"></property>
  10. </bean>
  11. </beans>

工厂方法模式

  在Spring中我们一般是将Bean的实例化直接交给容器去管理的,实现了使用和创建的分离,这时容器直接管理对象,还有种情况是,bean的创建过程我们交给一个工厂去实现,而Spring容器管理这个工厂。这个就是我们讲的工厂模式,在Spring中有两种实现一种是静态工厂方法模式,一种是动态工厂方法模式。以静态工厂来演示

  1. /**
  2. * User 工厂类
  3. * @author dpb[波波烤鸭]
  4. *
  5. */
  6. public class UserFactory {
  7. /**
  8. * 必须是static方法
  9. * @return
  10. */
  11. public static UserBean getInstance(){
  12. return new UserBean();
  13. }
  14. }

application.xml文件中注册

  1. <beans xmlns="http://www.springframework.org/schema/beans"
  2. xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
  3. xsi:schemaLocation="http://www.springframework.org/schema/beans
  4. http://www.springframework.org/schema/beans/spring-beans.xsd">
  5. <!-- 静态工厂方式配置 配置静态工厂及方法 -->
  6. <bean class="com.dpb.factory.UserFactory" factory-method="getInstance" id="user2"/>
  7. </beans>

6.适配器模式

  将一个类的接口转换成客户希望的另外一个接口。使得原本由于接口不兼容而不能一起工作的那些类可以在一起工作。这就是适配器模式。在Spring中在AOP实现中的Advice和interceptor之间的转换就是通过适配器模式实现的。

  1. class MethodBeforeAdviceAdapter implements AdvisorAdapter, Serializable {
  2. @Override
  3. public boolean supportsAdvice(Advice advice) {
  4. return (advice instanceof MethodBeforeAdvice);
  5. }
  6. @Override
  7. public MethodInterceptor getInterceptor(Advisor advisor) {
  8. MethodBeforeAdvice advice = (MethodBeforeAdvice) advisor.getAdvice();
  9. // 通知类型匹配对应的拦截器
  10. return new MethodBeforeAdviceInterceptor(advice);
  11. }
  12. }

详细介绍可以参考此文Spring之AOP适配器模式

7.装饰者模式

  装饰者模式又称为包装模式(Wrapper),作用是用来动态的为一个对象增加新的功能。装饰模式是一种用于代替继承的技术,无须通过继承增加子类就能扩展对象的新功能。使用对象的关联关系代替继承关系,更加灵活,同时避免类型体系的快速膨胀。

  spring中用到的包装器模式在类名上有两种表现:一种是类名中含有Wrapper,另一种是类名中含有Decorator。基本上都是动态地给一个对象添加一些额外的职责。

  具体的使用在Spring session框架中的SessionRepositoryRequestWrapper使用包装模式对原生的request的功能进行增强,可以将session中的数据和分布式数据库进行同步,这样即使当前tomcat崩溃,session中的数据也不会丢失。

查看需要的maven依赖

  1. <dependency>
  2. <groupId>org.springframework.session</groupId>
  3. <artifactId>spring-session</artifactId>
  4. <version>1.3.1.RELEASE</version>
  5. </dependency>

8.代理模式

  代理模式应该是大家非常熟悉的设计模式了,在Spring中AOP的实现中代理模式使用的很彻底,如果不了解代理模式欢迎查看我之前的文章,链接在顶部。

9.策略模式

  策略模式对应于解决某一个问题的一个算法族,允许用户从该算法族中任选一个算法解决某一问题,同时可以方便的更换算法或者增加新的算法。并且由客户端决定调用哪个算法,spring中在实例化对象的时候用到Strategy模式。

  1. @Override
  2. public Object instantiate(RootBeanDefinition bd, String beanName, BeanFactory owner) {
  3. // Don't override the class with CGLIB if no overrides.
  4. if (bd.getMethodOverrides().isEmpty()) {
  5. Constructor<?> constructorToUse;
  6. synchronized (bd.constructorArgumentLock) {
  7. constructorToUse = (Constructor<?>) bd.resolvedConstructorOrFactoryMethod;
  8. if (constructorToUse == null) {
  9. final Class<?> clazz = bd.getBeanClass();
  10. if (clazz.isInterface()) {
  11. throw new BeanInstantiationException(clazz, "Specified class is an interface");
  12. }
  13. try {
  14. if (System.getSecurityManager() != null) {
  15. constructorToUse = AccessController.doPrivileged(new PrivilegedExceptionAction<Constructor<?>>() {
  16. @Override
  17. public Constructor<?> run() throws Exception {
  18. return clazz.getDeclaredConstructor((Class[]) null);
  19. }
  20. });
  21. }
  22. else {
  23. constructorToUse = clazz.getDeclaredConstructor((Class[]) null);
  24. }
  25. bd.resolvedConstructorOrFactoryMethod = constructorToUse;
  26. }
  27. catch (Throwable ex) {
  28. throw new BeanInstantiationException(clazz, "No default constructor found", ex);
  29. }
  30. }
  31. }
  32. return BeanUtils.instantiateClass(constructorToUse);
  33. }
  34. else {
  35. // Must generate CGLIB subclass.
  36. return instantiateWithMethodInjection(bd, beanName, owner);
  37. }
  38. }

Spring中使用的设计模式的更多相关文章

  1. spring 中常用的设计模式

    一. Spring 中常见的设计模式 工厂模式 : BeanFactory 装饰器模式: BeanWrapper 代理模式: AopProxy 单例模式: ApplicationContext 委派模 ...

  2. Spring中常见的设计模式——代理模式

    一.代理模式的应用场景 生活中的中介,黄牛,等一系列帮助甲方做事的行为,都是代理模式的体现.代理模式(Proxy Pattern)是指为题对象提供一种代理,以控制对这个对象的访问.代理对象在客户端和目 ...

  3. Spring中常见的设计模式——适配器模式

    一.适配器模式的应用场景 适配器模式(Adapter Pattern)是指将一个类的接口转换成用户期待的另一个接口,使原本接口不兼容的类可以一起工作,属于构造设计模式. 适配器适用于以下几种业务场景: ...

  4. Spring中常见的设计模式——单例模式

    一.单例模式的应用场景 单例模式(singleton Pattern)是指确保一个类在任何情况下都绝对只有一个实例,并提供一个全局访问点.J2EE中的ServletContext,ServletCon ...

  5. Spring中常见的设计模式——委派模式

    一.委派模式的定义及应用场景 委派模式(Delegate Pattern)的基本作用是负责任务的调用和分配,跟代理模式很像,可以看做特殊情况下的静态的全权代理,但是代理模式注重过程,而委派模式注重结果 ...

  6. Spring中常见的设计模式——原型模式

    1.原型模式应用场景 当遇到大量耗费劳动力的 get,set赋值场景时,如下: public class SetGetParam { public void setParam(UserDto user ...

  7. Spring中常用的设计模式之:代理模式

    看了tom老师讲的深入分析spring源码,讲的挺好,做个小总结 代理模式的定义: 为其他对象提供一种代理以控制对这个对象的访问.在某些情况下,一个对象不适合或者不能直接引用另一个对象,而代理对象可以 ...

  8. Spring中常见的设计模式——策略模式

    策略模式(Strategy Pattern) 一.策略模式的应用场景 策略模式的应用场景如下: 系统中有很多类,而他们的区别仅仅在于行为不同. 一个系统需要动态的在集中算法中选择一种 二.用策略模式实 ...

  9. Spring中常见的设计模式——模板模式

    一.模板模式的应用场景 模板模式又叫模板方法模式(Template Method Pattern),指定义一个算法的骨架,并允许自雷为一个或者多个步骤提供实现.模板模式使得子类可以在不改变算法结果的情 ...

随机推荐

  1. 用js实现动态规划解决背包问题

    动态规划的原理: 移至到该同学的博文中,讲解的声动易懂 https://www.jianshu.com/p/a66d5ce49df5 现在主要是用js来实现动态规划 function bb(v, w, ...

  2. 使用Skaffold一键将项目发布到Kubernetes

    当前skaffold版本为v0.4,还未发布正式版本,不建议在生产环境中使用: skaffold用于开发人员快速部署程序到Kubernetes中:skaffold提供了dev.run两种模式:使用sk ...

  3. swust oj 1012

    哈希表(链地址法处理冲突) 1000(ms) 10000(kb) 2542 / 6517 采用除留余数法(H(key)=key %n)建立长度为n的哈希表,处理冲突用链地址法.建立链表的时候采用尾插法 ...

  4. Java作业五(2017-10-15)

    /*3-6.程序员;龚猛*/ 1 package zhenshu; import java.util.Scanner; public class text { public static void m ...

  5. SQL语句题

    SQL语句题 Student(Sno,Sname,Sage,Ssex)注释:学生表(学号,姓名,性别年龄,性别) Course(Cno,Cname,Tno) 注释:课程表(课程号,课程名称,教师编号) ...

  6. 吴恩达机器学习笔记54-开发与评价一个异常检测系统及其与监督学习的对比(Developing and Evaluating an Anomaly Detection System and the Comparison to Supervised Learning)

    一.开发与评价一个异常检测系统 异常检测算法是一个非监督学习算法,意味着我们无法根据结果变量

  7. [Swift]LeetCode60. 第k个排列 | Permutation Sequence

    The set [1,2,3,...,n] contains a total of n! unique permutations. By listing and labeling all of the ...

  8. 如何在 Linux 上复制文件/文件夹到远程系统?

    从一个服务器复制文件到另一个服务器,或者从本地到远程复制是 Linux 管理员的日常任务之一. 我觉得不会有人不同意,因为无论在哪里这都是你的日常操作之一.有很多办法都能处理这个任务,我们试着加以概括 ...

  9. python日志syslog运用

    syslog的官方说明在: https://docs.python.org/2/library/syslog.html#module-syslog 该模块的主要方式为: #!/usr/bin/pyth ...

  10. 11.Flask钩子函数

    在Flask中钩子函数是使用特定的装饰器的函数.为什么叫做钩子函数呢,是因为钩子函数可以在正常执行的代码中,插入一段自己想要执行的代码,那么这种函数就叫做钩子函数. before_first_requ ...