学习设计模式之代理模式

Posted 南淮北安

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了学习设计模式之代理模式相关的知识,希望对你有一定的参考价值。

一、定义


代理模式有点像老大和小弟,也有点像分销商。

主要解决的是问题是为某些资源的访问、对象的类的易用操作上提供方便使用的代理服务。而这种设计思想的模式经常会出现在我们的系统中,或者你用到过的组件中,它们都提供给你一种非常简单易用的方式控制原本你需要编写很多代码的进行使用的服务类。

类似这样的场景可以想到:

  • 你的数据库访问层面经常会提供一个较为基础的应用,以此来减少应用服务扩容时不至于数据库连接数暴增。
  • 使用过的一些中间件例如;RPC框架,在拿到jar包对接口的描述后,中间件会在服务启动的时候生成对应的代理类,当调用接口的时候,实际是通过代理类发出的socket信息进行通过。
  • 另外像我们常用的MyBatis,基本是定义接口但是不需要写实现类,就可以对xml或者自定义注解里的sql语句进行增删改查操作

二、问题背景


在本案例中我们模拟实现mybatis-spring中代理类生成部分

对于Mybatis的使用中只需要定义接口不需要写实现类就可以完成增删改查操作,有疑问的小伙伴,在本章节中就可以学习到这部分知识。解析下来我们会通过实现一个这样的代理类交给spring管理的核心过程,来讲述代理类模式。

这样的案例场景在实际的业务开发中其实不多,因为这是将这种思想运用在中间件开发上,而很多小伙伴经常是做业务开发,所以对Spring的bean定义以及注册和对代理以及反射调用的知识了解的相对较少。但可以通过本章节作为一个入门学习,逐步了解

四、代理类模式实现过程

接下来会使用代理类模式来模拟实现一个Mybatis中对类的代理过程,也就是只需要定义接口,就可以关联到方法注解中的sql语句完成对数据库的操作

这里需要注意一些知识点:

  • BeanDefinitionRegistryPostProcessor,spring的接口类用于处理对bean的定义注册。
  • GenericBeanDefinition,定义bean的信息,在mybatis-spring中使用到的是;
  • ScannedGenericBeanDefinition 略有不同。
  • FactoryBean,用于处理bean工厂的类,这个类非常见。

1. 工程结构

itstack-demo-design-12-00
└── src
    ├── main
    │   ├── java
    │   │   └── org.itstack.demo.design
    │   │       ├── agent
    │   │       │	├── MapperFactoryBean.java
    │   │       │	├── RegisterBeanFactory.java
    │   │       │	└── Select.java
    │   │       └── IUserDao.java
    │   └── resources	
    │       └── spring-config.xml
    └── test
        └── java
            └── org.itstack.demo.test
                └── ApiTest.java

代理模式中间件模型结构

此模型中涉及的类并不多,但都是抽离出来的核心处理类。主要的事情就是对类的代理和注册到spring中。

上图中最上面是关于中间件的实现部分,下面对应的是功能的使用

2. 代码实现

(1)自定义注解

@Documented
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.METHOD)
public @interface Select 

    String value() default "";  // sql语句



这里我们定义了一个模拟mybatis-spring中的自定义注解,用于使用在方法层面。

(2)Dao层接口

public interface IUserDao 

    @Select("select userName from user where id = #uId")
    String queryUserInfo(String uId);


这里定义一个Dao层接口,并把自定义注解添加上。这与你使用的mybatis组件是一样的。
1和2是我们的准备工作,后面开始实现中间件功能部分

(3)代理类定义

public class MapperFactoryBean<T> implements FactoryBean<T> 

    private Logger logger = LoggerFactory.getLogger(MapperFactoryBean.class);

    private Class<T> mapperInterface;

    public MapperFactoryBean(Class<T> mapperInterface) 
        this.mapperInterface = mapperInterface;
    

    @Override
    public T getObject() throws Exception 
        InvocationHandler handler = (proxy, method, args) -> 
            Select select = method.getAnnotation(Select.class);
            logger.info("SQL:", select.value().replace("#uId", args[0].toString()));
            return args[0] + ",小傅哥,bugstack.cn - 沉淀、分享、成长,让自己和他人都能有所收获!";
        ;
        return (T) Proxy.newProxyInstance(this.getClass().getClassLoader(), new Class[]mapperInterface, handler);
    

    @Override
    public Class<?> getObjectType() 
        return mapperInterface;
    

    @Override
    public boolean isSingleton() 
        return true;
    



(4)将Bean定义注册到Spring容器

public class RegisterBeanFactory implements BeanDefinitionRegistryPostProcessor 
    
    @Override
    public void postProcessBeanDefinitionRegistry(BeanDefinitionRegistry registry) throws BeansException 
        
        GenericBeanDefinition beanDefinition = new GenericBeanDefinition();
        beanDefinition.setBeanClass(MapperFactoryBean.class);
        beanDefinition.setScope("singleton");
        beanDefinition.getConstructorArgumentValues().addGenericArgumentValue(IUserDao.class);

        BeanDefinitionHolder definitionHolder = new BeanDefinitionHolder(beanDefinition, "userDao");
        BeanDefinitionReaderUtils.registerBeanDefinition(definitionHolder, registry);
    

    @Override
    public void postProcessBeanFactory(ConfigurableListableBeanFactory configurableListableBeanFactory) throws BeansException 
        // left intentionally blank
    



这里我们将代理的bean交给spring容器管理,也就可以非常方便让我们可以获取到代理的bean。这部分是spring中关于一个bean注册过程的源码。

GenericBeanDefinition,用于定义一个bean的基本信息 setBeanClass(MapperFactoryBean.class);,

也包括可以透传给构造函数信息addGenericArgumentValue(IUserDao.class);

最后使用 BeanDefinitionReaderUtils.registerBeanDefinition,进行bean的注册,也就是注册到DefaultListableBeanFactory

(5)配置文件spring-config

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-3.0.xsd"
       default-autowire="byName">

    <bean id="userDao" class="org.itstack.demo.design.agent.RegisterBeanFactory"/>

</beans>

接下来在配置文件中添加我们的bean配置,在mybatis的使用中一般会配置扫描的dao层包,这样就可以减少这部分的配置

3. 测试验证

@Test
public void test_IUserDao() 
    BeanFactory beanFactory = new ClassPathXmlApplicationContext("spring-config.xml");
    IUserDao userDao = beanFactory.getBean("userDao", IUserDao.class);
    String res = userDao.queryUserInfo("100001");
    logger.info("测试结果:", res);


测试的过程比较简单,通过加载Bean工厂获取我们的代理类的实例对象,之后调用方法返回结果。

那么这个过程你可以看到我们是没有对接口先一个实现类的,而是使用代理的方式给接口生成一个实现类,并交给spring管理

五、总结

关于这部分代理模式的讲解我们采用了开发一个关于mybatis-spring中间件中部分核心功能来体现代理模式的强大之处,所以涉及到了一些关于代理类的创建以及spring中bean的注册这些知识点,可能在平常的业务开发中都是很少用到的,但是在中间件开发中确实非常常见的操作。

代理模式除了开发中间件外还可以是对服务的包装,物联网组件等等,让复杂的各项服务变为轻量级调用、缓存使用。你可以理解为你家里的电灯开关,我们不能操作220v电线的人肉连接,但是可以使用开关,避免触电。

代理模式的设计方式可以让代码更加整洁、干净易于维护,虽然在这部分开发中额外增加了很多类也包括了自己处理bean的注册等,但是这样的中间件复用性极高也更加智能,可以非常方便的扩展到各个服务应用中。

以上是关于学习设计模式之代理模式的主要内容,如果未能解决你的问题,请参考以下文章

23天设计模式之代理模式

代理设计模式之静态代理与动态代理(超..)详解

入门设计模式之代理

设计模式学习笔记之代理模式

Java进阶篇设计模式之七 ----- 享元模式和代理模式

Java学习笔记——设计模式之四.代理模式