个人随笔
Spring框架中的设计模式
2020-09-11 22:16:53

Spring的核心功能是IOC容器以及AOP面向切面编程,同样也是很多Web后端工程师每天都要打交道的框架,

1.简单工厂模式

考虑这样一个场景:当A对象需要调用B对象的方法时,需要在A中new一个B的实例,把这种方式叫作硬编码耦合,它的缺点是一旦需求发生变化,比如需要使用C类来代替B时,就要改写A类的方法。假如应用中有1000个类以硬编码的方式耦合了B,那改起来就费劲了。于是简单工厂模式就登场了,简单工厂模式又叫静态工厂方法,其实质是由一个工厂类根据传入的参数,动态决定应该创建哪一个产品类。

Spring中的BeanFactory就是简单工厂模式的体现,BeanFactory是Spring IOC容器中的一个核心接口,它的定义如下:

  1. public interface BeanFactory {
  2. Object getBean(String name) throws BeansException;
  3. <T> T getBean(String name, Class<T> requiredType);
  4. Object getBean(String name, Object... args);
  5. <T> T getBean(Class<T> requiredType);
  6. <T> T getBean(Class<T> requiredType, Object... args);
  7. boolean containsBean(String name);
  8. boolean isSingleton(String name);
  9. boolea isPrototype(String name);
  10. boolean isTypeMatch(String name, ResolvableType typeToMatch);
  11. boolean isTypeMatch(String name, Class<?> typeToMatch);
  12. Class<?> getType(String name);
  13. String[] getAliases(String name);
  14. }

可以通过它的具体实现类(比如ClassPathXmlApplicationContext)来获取Bean:

  1. BeanFactory bf = new ClassPathXmlApplicationContext("spring.xml");
  2. User userBean = (User) bf.getBean("userBean");

从上面代码可以看到,使用者不需要自己来new对象,而是通过工厂类的方法getBean来获取对象实例,这是典型的简单工厂模式,只不过Spring是用反射机制来创建Bean的。

2.工厂方法模式

工厂方法模式说白了其实就是简单工厂模式的一种升级或者说是进一步抽象,它可以应用于更加复杂的场景,灵活性也更高。在简单工厂中,由工厂类进行所有的逻辑判断、实例创建;如果不想在工厂类中进行判断,可以为不同的产品提供不同的工厂,不同的工厂生产不同的产品,每一个工厂都只对应一个相应的对象,这就是工厂方法模式。

Spring中的FactoryBean就是这种思想的体现,FactoryBean可以理解为工厂Bean,先来看看它的定义:

  1. public interface FactoryBean<T> {
  2. T getObject();
  3. Class<?> getObjectType();
  4. boolean isSingleton();
  5. }

定义一个类UserFactoryBean来实现FactoryBean接口,主要是在getObject方法里new一个User对象。这样通过getBean(id) 获得的是该工厂所产生的User的实例,而不是UserFactoryBean本身的实例,像下面这样:

  1. BeanFactory bf = new ClassPathXmlApplicationContext("user.xml");
  2. User userBean = (User) bf.getBean("userFactoryBean");

3.单例模式

单例模式是指一个类在整个系统运行过程中,只允许产生一个实例。在Spring中,Bean可以被定义为两种模式:Prototype(多例)和Singleton(单例),Spring Bean默认是单例模式。那Spring是如何实现单例模式的呢?答案是通过单例注册表的方式,具体来说就是使用了HashMap。对代码进行了简化:

  1. public class DefaultSingletonBeanRegistry {
  2. //使用了线程安全容器ConcurrentHashMap,保存各种单实例对象
  3. private final Map<String, Object> singletonObjects = new ConcurrentHashMap<String, Object>;
  4. protected Object getSingleton(String beanName) {
  5. //先到HashMap中拿Object
  6. Object singletonObject = singletonObjects.get(beanName);
  7. //如果没拿到通过反射创建一个对象实例,并添加到HashMap中
  8. if (singletonObject == null) {
  9. singletonObjects.put(beanName,
  10. Class.forName(beanName).newInstance());
  11. }
  12. //返回对象实例
  13. return singletonObjects.get(beanName);
  14. }
  15. }

上面的代码逻辑比较清晰,先到HashMap去拿单实例对象,没拿到就创建一个添加到HashMap。

4.代理模式

所谓代理,是指它与被代理对象实现了相同的接口,客户端必须通过代理才能与被代理的目标类进行交互,而代理一般在交互的过程中(交互前后),进行某些特定的处理,比如在调用这个方法前做前置处理,调用这个方法后做后置处理。代理模式中有下面几种角色:

抽象接口:定义目标类及代理类的共同接口,这样在任何可以使用目标对象的地方都可以使用代理对象。
目标对象: 定义了代理对象所代表的目标对象,专注于业务功能的实现。
代理对象: 代理对象内部含有目标对象的引用,收到客户端的调用请求时,代理对象通常不会直接调用目标对象的方法,而是在调用之前和之后实现一些额外的逻辑。
代理模式的好处是,可以在目标对象业务功能的基础上添加一些公共的逻辑,比如我们想给目标对象加入日志、权限管理和事务控制等功能,就可以使用代理类来完成,而没必要修改目标类,从而使得目标类保持稳定。这其实是开闭原则的体现,不要随意去修改别人已经写好的代码或者方法。

代理又分为静态代理和动态代理两种方式。静态代理需要定义接口,被代理对象(目标对象)与代理对象(Proxy)一起实现相同的接口,我们通过一个例子来理解一下:

  1. //抽象接口
  2. public interface IStudentDao {
  3. void save();
  4. }
  5. //目标对象
  6. public class StudentDao implements IStudentDao {
  7. public void save() {
  8. System.out.println("保存成功");
  9. }
  10. }
  11. //代理对象
  12. public class StudentDaoProxy implements IStudentDao{
  13. //持有目标对象的引用
  14. private IStudentDao target;
  15. public StudentDaoProxy(IStudentDao target){
  16. this.target = target;
  17. }
  18. //在目标功能对象方法的前后加入事务控制
  19. public void save() {
  20. System.out.println("开始事务");
  21. target.save();//执行目标对象的方法
  22. System.out.println("提交事务");
  23. }
  24. }
  25. public static void main(String[] args) {
  26. //创建目标对象
  27. StudentDao target = new StudentDao();
  28. //创建代理对象,把目标对象传给代理对象,建立代理关系
  29. StudentDaoProxy proxy = new StudentDaoProxy(target);
  30. //执行的是代理的方法
  31. proxy.save();
  32. }

而Spring的AOP采用的是动态代理的方式,而动态代理就是指代理类在程序运行时由JVM动态创建。在上面静态代理的例子中,代理类(StudentDaoProxy)是我们自己定义好的,在程序运行之前就已经编译完成。而动态代理,代理类并不是在Java代码中定义的,而是在运行时根据我们在Java代码中的“指示”动态生成的。那我们怎么“指示”JDK去动态地生成代理类呢?

在Java的java.lang.reflect包里提供了一个Proxy类和一个InvocationHandler接口,通过这个类和这个接口可以生成动态代理对象。具体来说有如下步骤:

step1.定义一个InvocationHandler类,将需要扩展的逻辑集中放到这个类中,比如下面的例子模拟了添加事务控制的逻辑。
  1. public class MyInvocationHandler implements InvocationHandler {
  2. private Object obj;
  3. public MyInvocationHandler(Object obj){
  4. this.obj=obj;
  5. }
  6. @Override
  7. public Object invoke(Object proxy, Method method, Object[] args)
  8. throws Throwable {
  9. System.out.println("开始事务");
  10. Object result = method.invoke(obj, args);
  11. System.out.println("开始事务");
  12. return result;
  13. }
  14. }
step2.使用Proxy的newProxyInstance方法动态的创建代理对象:
  1. public static void main(String[] args) {
  2. //创建目标对象StudentDao
  3. IStudentDao stuDAO = new StudentDao();
  4. //创建MyInvocationHandler对象
  5. InvocationHandler handler = new MyInvocationHandler(stuDAO);
  6. //使用Proxy.newProxyInstance动态的创建代理对象stuProxy
  7. IStudentDao stuProxy = (IStudentDao)
  8. Proxy.newProxyInstance(stuDAO.getClass().getClassLoader(), stuDAO.getClass().getInterfaces(), handler);
  9. //动用代理对象的方法
  10. stuProxy.save();
  11. }

上面的代码实现和静态代理一样的功能,相比于静态代理,动态代理的优势在于可以很方便地对代理类的函数进行统一的处理,而不用修改每个代理类中的方法。

Spring实现了通过动态代理对类进行方法级别的切面增强,解释一下这句话,其实就是动态生成目标对象的代理类,并在代理类的方法中设置拦截器,通过执行拦截器中的逻辑增强了代理方法的功能,从而实现AOP。

 17

啊!这个可能是世界上最丑的留言输入框功能~


当然,也是最丑的留言列表

有疑问发邮件到 : suibibk@qq.com 侵权立删
Copyright : 个人随笔   备案号 : 粤ICP备18099399号-2