Spring用到了哪些设计模式?

Spring框架中广泛运用了多种设计模式,以实现其强大的功能和灵活性。

以下是一些Spring框架中常见的设计模式:

1. 工厂模式(Factory Pattern)

  • • 用途:Spring通过BeanFactory和ApplicationContext等接口创建并管理对象实例。这种方式将对象的创建与使用解耦,使得程序更加灵活和可扩展。

  • • 实现:BeanFactory是一个工厂接口,提供了获取Bean对象的方法(如getBean(String name))。ApplicationContext是BeanFactory的子接口,提供了更多高级功能,如国际化、事件传播和自动Bean注入等。

2. 单例模式(Singleton Pattern)

  • • 用途:Spring框架中的Bean默认是单例的,即在容器中只有一个实例。这种单例模式的设计有助于节省资源并提高性能。

  • • 实现:当Spring容器启动时,会为每个作用域为singleton的Bean创建并维护一个单实例对象,这些对象会被存储在一个缓存中,从而确保每次注入时都是同一个实例。

3. 代理模式(Proxy Pattern)

  • • 用途:Spring的AOP(面向切面编程)功能大量使用了代理模式。AOP通过在目标方法执行前后添加额外的行为(如日志、事务管理等),而这些额外的行为是通过代理对象来实现的。

  • • 实现:Spring提供了两种代理方式:JDK动态代理和CGLIB代理。如果目标类实现了一个接口,Spring默认使用JDK动态代理;如果没有实现接口,则使用CGLIB来生成子类代理。

4. 模板方法模式(Template Method Pattern)

  • • 用途:用于定义一个操作的算法骨架,将一些步骤推迟到子类中去实现。Spring框架在JDBC、Hibernate、JPA、事务管理等模块中都使用了模板方法模式。

  • • 实现:例如,JdbcTemplate类使用模板方法模式来封装与数据库交互的步骤。开发者只需提供SQL语句和参数,而不必关心资源获取、异常处理和资源释放等细节。

5. 观察者模式(Observer Pattern)

  • • 用途:Spring框架中的事件驱动机制使用了观察者模式。通过观察者模式,Spring框架实现了事件的发布和订阅机制,使得组件之间可以更加灵活地进行通信和协作。

  • • 实现:ApplicationEventPublisher是一个事件发布者,ApplicationListener是一个事件监听器。Spring容器允许多个监听器订阅和监听特定类型的事件,当事件发生时,所有订阅的监听器都会收到通知。

6. 策略模式(Strategy Pattern)

  • • 用途:用于定义一系列算法,将每一个算法封装起来,并让它们可以相互替换。策略模式让算法独立于使用它的客户而变化,提高了代码的可拓展性,降低了耦合度。

  • • 实现:Spring框架在很多地方使用策略模式,例如在事务管理中使用不同的事务管理策略(如JDBC、JTA),在视图解析器(ViewResolver)中使用不同的视图解析策略(如JSP、Thymeleaf)。

7. 适配器模式(Adapter Pattern)

  • • 用途:用于将一个接口转换为客户希望的另一个接口。Spring中的适配器模式在AOP和MVC框架中都有体现。

  • • 实现:在Spring MVC中,HandlerAdapter用于将不同的处理器(Handler)适配为统一的接口。例如,HttpRequestHandlerAdapter和SimpleControllerHandlerAdapter分别适配HttpRequestHandler和Controller,使得Spring MVC可以支持多种不同类型的控制器。

8. 装饰器模式(Decorator Pattern)

  • • 用途:用于动态地为对象添加行为而不改变其结构。Spring使用装饰器模式来增强Bean的功能。

  • • 实现:在Spring AOP中,代理对象(代理类)就是对目标对象的增强(装饰),可以动态地为目标对象添加新的行为(如方法拦截、日志记录、事务管理等)。

9. 依赖注入和控制反转(Dependency Injection and Inversion of Control)

  • • 用途:依赖注入(DI)模式用于将对象的依赖关系从内部转移到外部,使得对象更加解耦并且易于测试。控制反转(IoC)则是指将对象的创建和依赖关系的管理交给Spring容器,从而实现了对象的解耦。

  • • 实现:Spring通过XML配置、注解或Java配置类等方式,自动为Bean提供它们所需的依赖项。依赖注入主要有两种方式:构造函数注入和setter方法注入。

示例演示

1. 工厂模式(Factory Pattern)

// 定义一个Bean接口
public interface MyBean {
    void doSomething();
}

// 实现Bean接口
public class MyBeanImpl implements MyBean {
    @Override
    public void doSomething() {
        System.out.println("Doing something...");
    }
}

// 配置Spring的BeanFactory
// Spring配置文件(XML方式)
<beans ...>
    <bean id="myBean" class="com.example.MyBeanImpl" />
</beans>

// 使用BeanFactory获取Bean实例
BeanFactory factory = new XmlBeanFactory(new ClassPathResource("spring-config.xml"));
MyBean myBean = (MyBean) factory.getBean("myBean");
myBean.doSomething();

注意XmlBeanFactory 在Spring 3.1后已被废弃,建议使用ApplicationContext

2. 单例模式(Singleton Pattern)

Spring默认Bean是单例的,所以只需配置一个Bean即可。

<!-- Spring配置文件 -->
<beans ...>
    <bean id="mySingletonBean" class="com.example.MySingletonBean" scope="singleton" />
</beans>

3. 代理模式(Proxy Pattern)

使用Spring AOP实现代理模式:

// 定义一个服务接口
public interface MyService {
    void performTask();
}

// 实现服务接口
public class MyServiceImpl implements MyService {
    @Override
    public void performTask() {
        System.out.println("Performing task...");
    }
}

// 定义一个切面(Aspect)
@Aspect
public class MyAspect {
    @Before("execution(* com.example.MyService.*(..))")
    public void beforeAdvice(JoinPoint joinPoint) {
        System.out.println("Before method: " + joinPoint.getSignature());
    }
}

// Spring配置文件(启用AOP)
<beans ...>
    <aop:aspectj-autoproxy />
    <bean id="myService" class="com.example.MyServiceImpl" />
    <bean id="myAspect" class="com.example.MyAspect" />
</beans>

// 使用代理对象
ApplicationContext context = new ClassPathXmlApplicationContext("spring-config.xml");
MyService myService = (MyService) context.getBean("myService");
myService.performTask();  // 执行时会触发切面中的beforeAdvice方法

4. 模板方法模式(Template Method Pattern)

Spring的JdbcTemplate是一个典型的模板方法模式实现:

// 使用JdbcTemplate进行查询
JdbcTemplate jdbcTemplate = new JdbcTemplate(dataSource);
String sql = "SELECT * FROM my_table WHERE id = ?";
MyEntity entity = jdbcTemplate.queryForObject(sql, new BeanPropertyRowMapper<>(MyEntity.class), id);

5. 观察者模式(Observer Pattern)

使用Spring的事件机制:

// 定义一个事件
public class MyEvent extends ApplicationEvent {
    public MyEvent(Object source) {
        super(source);
    }
}

// 定义一个事件监听器
@Component
public class MyEventListener implements ApplicationListener<MyEvent> {
    @Override
    public void onApplicationEvent(MyEvent event) {
        System.out.println("Received event: " + event);
    }
}

// 发布事件
ApplicationContext context = ...;
context.publishEvent(new MyEvent(this));

6. 策略模式(Strategy Pattern)

在Spring中配置不同的策略实现:

// 定义一个策略接口
public interface MyStrategy {
    void execute();
}

// 实现策略接口
public class StrategyA implements MyStrategy {
    @Override
    public void execute() {
        System.out.println("Executing Strategy A");
    }
}

public class StrategyB implements MyStrategy {
    @Override
    public void execute() {
        System.out.println("Executing Strategy B");
    }
}

// Spring配置文件
<beans ...>
    <bean id="strategyA" class="com.example.StrategyA" />
    <bean id="strategyB" class="com.example.StrategyB" />
</beans>

// 使用策略
ApplicationContext context = ...;
MyStrategy strategy = (MyStrategy) context.getBean("strategyA");
strategy.execute();

7. 适配器模式(Adapter Pattern)

在Spring MVC中使用适配器:

Spring MVC已经内置了多个适配器,如HttpRequestHandlerAdapterSimpleControllerHandlerAdapter。你通常不需要自己实现适配器,除非有特殊需求。

8. 装饰器模式(Decorator Pattern)

在Spring AOP中,代理对象可以看作是对目标对象的装饰:

上面的代理模式示例中,MyAspect就是对MyService的装饰。

9. 依赖注入和控制反转(Dependency Injection and Inversion of Control)

使用Spring的依赖注入:

// 定义一个依赖类
public class MyDependency {
    public void someMethod() {
        System.out.println("Dependency method");
    }
}

// 定义一个使用依赖的类
public class MyServiceWithDependency {
    private MyDependency dependency;

    // 使用setter注入
    public void setDependency(MyDependency dependency) {
        this.dependency = dependency;
    }

    public void perform() {
        dependency.someMethod();
        System.out.println("Service performing...");
    }
}

// Spring配置文件
<beans ...>
    <bean id="myDependency" class="com.example.MyDependency" />
    <bean id="myServiceWithDependency" class="com.example.MyServiceWithDependency">
        <property name="dependency" ref="myDependency" />
    </bean>
</beans>

// 使用依赖注入的对象
ApplicationContext context = new ClassPathXmlApplicationContext("spring-config.xml");
MyServiceWithDependency service = (MyServiceWithDependency) context.getBean("myServiceWithDependency");
service.perform();
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

这孩子叫逆

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值