动态代理的实际应用

简介: 最近在用 Python 的 SQLAlchemy 库时(一个类似于 Hibernate 的 ORM 框架),发现它的 Events 事件还挺好用。简单说就是当某张表的数据发生变化(曾、删、改)时会有一个事件回调,这样一些埋点之类的需求都可以实现在这里,同时和业务代码完全解耦,维护起来也很方便。

前言


最近在用 PythonSQLAlchemy 库时(一个类似于 HibernateORM 框架),发现它的 Events 事件还挺好用。


简单说就是当某张表的数据发生变化(曾、删、改)时会有一个事件回调,这样一些埋点之类的需求都可以实现在这里,同时和业务代码完全解耦,维护起来也很方便。


例如当订单状态发生变化需要发异步通知这样的需求也可以利用这个实现。


根据我之前使用 Mybatis 的经验,好像没怎么注意有这个功能,查阅了下发现 Hibernate 是支持的,只是我用得也少,所以也没怎么在意。


说这些的主要原因是我打算为之前写的 cicada (轻量的 http 框架)加一个数据库操作包,也实现类似的功能。


示例


最终的使用效果如下:


第一版本还比较粗糙,但功能都具备。



第一步:需要实现一个初始化接口,该接口会在应用初始化的时候执行。


紧接着我们需要定义一个 Model


@Data
@OriginName("user")
@ToString
public class User extends Model {
    @PrimaryId
    private Integer id ;
    private String name ;
    private String password ;
    @FieldName(value = "city_id")
    private Integer cityId ;
    private String description ;
}


它所对应的表结构如下:


CREATE TABLE `user` (
  `id` int(11) NOT NULL AUTO_INCREMENT,
  `name` varchar(50) DEFAULT NULL,
  `password` varchar(100) DEFAULT NULL,
  `description` varchar(100) DEFAULT NULL,
  `roleId` int(11) DEFAULT NULL COMMENT '角色ID',
  `city_id` int(11) DEFAULT NULL,
  PRIMARY KEY (`id`)
)


当需要查询数据时:



便可以这样访问数据库。


当需要更新数据时:



在初始化 DBHandle 时指定一个回调接口(也就是这里的 UserUpdateListener),便可以在修改数据的时候拿到本次修改的数据实体。


@Slf4j
public class UserUpdateListener implements DataChangeListener {
    @Override
    public void listener(Object obj) {
        log.info("user update data={}", obj.toString());
    }
}


同时我们可以在控制台看到数据修改时的回调结果:



这样就实现了文初所提到的功能,便可以实现一些数据变化后需要执行的业务逻辑。


实现


下面重点来看看这个功能的实现过程;其实通过生成 DBHandle(数据库增删改的接口)实例的 API 便可以看出些端倪。


DBHandle handle = (DBHandle) new HandleProxy(DBHandle.class).getInstance(new UserSaveListener());


DBHandel 虽然是个接口,但是它并不是使用一个实现类来实现的,而是通过代理生成。


那通过代理生成比直接实例化实现类有啥好处呢?


举个例子,比如现在你想买一个新手机。



第一种方式可以直接在官方旗舰店买一个标配的手机,没有额外的东西只有一个手机。


当然你也可以在某些第三方经销商那里购买带套餐的,比如套餐一在标配的基础上多了保护壳、贴膜之类的附加属性。


这个经销商就类似于我们这里的代理类,他可以在原有实现的基础上新增一些东西,至于新增什么全看你自己的需要了。


而之所以叫动态代理,也是因为这个代理类是在程序运行过程中动态创建的,在编译过程中并不能确定这个类的全限定名。


下面来看看这个代理类是如何生成的:



主要利用 JDK 自带的 API 实现的,具体参数可以直接参考官方文档:

docs.oracle.com/javase/8/do…


总之这样便可以创建一个 DBHandler 接口的代理对象,而真正的代理过程是在

InvocationHandler#invoke() 函数中实现的:



这里的实现也是非常简单,在实现完代理对象的业务逻辑后便回调我们传入的事件接口,其中的参数便是当前的数据库 Model 实体对象。


不过需要注意的是,这个事件回调和业务线程是同一个,所以写在这里的逻辑建议都为异步(Hibernate 和 SQLAlchemy 都存在这个情况)。


总结


以上便是整个动态代理实现 ORM 监听机制的全过程,其实可以看出并没有它名称那样看起来高大上,当然本身实现也比较简单。


同时也不止这一种实现方式,例如:


  • cglib


  • javassist


  • ASM


etc..


他们的具体实现及优劣就不在本文探讨了,感兴趣的后续我会将这个功能用这几种方式实现一遍。


同时动态代理的应用也不止于此,比如:


  • RPC 中无感知的远程调用。


  • Spring 中的 AOP、拦截器等。


后续会继续完善这个 ORM 库,甚至可以独立出来作为一个小巧的数据库工具也未尝不可。


相关源码见此处:github.com/TogetherOS/…


相关文章
|
Java 程序员
动态代理
动态代理
51 0
|
6月前
|
设计模式 Java
动态代理详解
【2月更文挑战第7天】
动态代理详解
jdk动态代理和cglb动态代理
jdk动态代理和cglb动态代理
|
Java Spring
jdk动态代理和cglib动态代理
只有聪明人才能看见的简介~( ̄▽ ̄~)~
87 0
jdk动态代理和cglib动态代理
|
存储 Java Apache
一文理解动态代理和静态代理
一文理解动态代理和静态代理
200 0
一文理解动态代理和静态代理
|
Java 编译器 Maven
动态代理竟然如此简单!(二)
这篇文章我们来聊一下 Java 中的动态代理。 动态代理在 Java 中有着广泛的应用,比如 AOP 的实现原理、RPC远程调用、Java 注解对象获取、日志框架、全局性异常处理、事务处理等。
97 0
动态代理竟然如此简单!(二)
|
设计模式 Java 程序员
动态代理竟然如此简单!(一)
这篇文章我们来聊一下 Java 中的动态代理。 动态代理在 Java 中有着广泛的应用,比如 AOP 的实现原理、RPC远程调用、Java 注解对象获取、日志框架、全局性异常处理、事务处理等。
151 0
动态代理竟然如此简单!(一)
深入理解动态代理
代理模式是在不修改原有代码逻辑的情况下,对原有代码逻辑增强的一种方式,要了解什么是动态代理,首先要知道什么是静态代理。
深入理解动态代理
|
设计模式 Java API
静态代理、动态代理(JDK动态代理,Cglib动态代理)(1)
静态代理、动态代理(JDK动态代理,Cglib动态代理)(1)
129 0
静态代理、动态代理(JDK动态代理,Cglib动态代理)(1)
|
Java Spring
静态代理与JDK动态代理与CGLIB动态代理(上)
静态代理与JDK动态代理与CGLIB动态代理(上)