动态代理的实际应用

简介: 最近在用 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/…


相关文章
|
弹性计算 缓存 负载均衡
ECS性能优化建议
ECS性能优化建议
681 3
|
机器学习/深度学习 数据可视化 PyTorch
【PyTorch】TensorBoard基本使用
【PyTorch】TensorBoard基本使用
1031 0
|
存储
C++-静态局部变量
C++-静态局部变量
106 0
|
JSON 监控 安全
JSONP 被劫持后会对用户造成哪些危害?
JSONP 被劫持后可能对用户造成严重的危害,涉及信息安全、财产安全和隐私保护等多个方面。因此,在使用 JSONP 进行跨域数据交互时,必须采取有效的安全措施来防止劫持事件的发生。
303 7
|
存储
RAID与LVM
RAID与LVM
554 1
|
机器学习/深度学习 数据采集 算法
探索机器学习在金融风控中的应用与挑战
【8月更文挑战第10天】随着金融科技的迅速发展,机器学习技术被广泛应用于金融服务领域,尤其是风险控制。本文深入探讨了机器学习在金融风控中的角色,分析了其在信用评分、欺诈检测等方面的应用,并指出了实施过程中面临的数据质量、模型解释性、法规遵从等挑战。文章旨在为金融机构提供机器学习应用的参考框架和应对策略,以增强风险管理能力。
194 7
|
移动开发 JavaScript 前端开发
javascript监听浏览器离开、进入行为
javascript监听浏览器离开、进入行为
601 0
|
存储 持续交付 Docker
云效流水线问题之推送镜像报443连接超时如何解决
云效镜像是指存储在阿里云效服务中的容器镜像,它们可以用于持续集成和持续部署(CI/CD)流程中;本合集将介绍如何在云效平台上管理和使用镜像资源,以及常见的镜像问题和解决办法。
320 0
|
机器学习/深度学习 数据采集 运维
数据挖掘实践(金融风控):金融风控之贷款违约预测挑战赛(上篇)[xgboots/lightgbm/Catboost等模型]–模型融合:stacking、blending
数据挖掘实践(金融风控):金融风控之贷款违约预测挑战赛(上篇)[xgboots/lightgbm/Catboost等模型]–模型融合:stacking、blending
|
前端开发
css:flex布局下overflow失效
css:flex布局下overflow失效
504 0