「这是我参与11月更文挑战的第23天,活动详情查看:2021最后一次更文挑战」
本文被《Spring Boot 实战》专栏收录。
你好,我是看山。
在分布式系统中,我们会需要 ID 生成器的组件,这个组件可以实现帮助我们生成顺序的或者带业务含义的 ID。
目前有很多经典的 ID 生成方式,比如数据库自增列(自增主键或序列)、Snowflake 算法、美团 Leaf 算法等等,所以,会有一些公司级或者业务级的 ID 生成器组件的诞生。本文就是通过 BeanPostProcessor 实现动态注入 ID 生成器的实战。
在 Spring 中,实现注入的方式很多,比如 springboot 的 starter,在自定义的 Configuration 中初始化 ID 生成器的 Bean,业务代码中通过@AutoWired
或者@Resource
注入即可,开箱即用。这种方式简单直接,但是缺点也是过于简单,缺少了使用方自定义的入口。
考虑一下实际场景,在同一个业务单据中,要保持 ID 的唯一,但是在不同单据中,可以重复。而且,这些算法在生成 ID 的时候,为了保持多线程返回结果唯一,都会锁定共享资源。如果不同业务,并发情景不同,可能低并发的业务被高并发的业务阻塞获取 ID,造成一些性能的损失。所以,我们要考虑将 ID 生成器,根据业务隔离开,这样 springboot 的 starter 就会显得不够灵活了。
实现
根据上面的需求,我们可以分几步实现我们的逻辑:
- 自定义属性注解,用于判断是否需要注入属性对象
- 定义 ID 生成器接口、实现类,以及工厂类,工厂类是为了根据定义创建不同的 ID 生成器实现对象
- 定义 BeanPostProcessor,查找使用自定义注解定义的属性,实现注入
自定义注解
首先自定义一个注解,可以定义一个value
属性,作为隔离业务的标识:
1 | java复制代码@Retention(RetentionPolicy.RUNTIME) |
定义 ID 生成器
定义 ID 生成器的接口:
1 | java复制代码public interface IdGenerator { |
实现 ID 生成器接口,偷懒使用AtomicLong
实现自增,同时考虑 ID 生成器是分组的,通过ConcurrentHashMap
实现 ID 生成器的持有:
1 | java复制代码class DefaultIdGenerator implements IdGenerator { |
如前面设计的,我们需要一个工厂类来创建 ID 生成器,示例中使用最简单的实现,我们真正使用的时候,还可以通过更加灵活的 SPI 实现(关于 SPI 的实现,这里挖个坑,后面专门写一篇填坑):
1 | java复制代码public enum IdGeneratorFactory { |
定义 BeanPostProcessor
前面都是属于基本操作,这里才是扩展的核心。我们的实现逻辑是:
- 扫描 bean 的所有属性,然后找到定义了
IdGeneratorClient
注解的属性 - 获取注解的
value
值,作为 ID 生成器的分组标识 - 使用
IdGeneratorFactory
这个工厂类生成 ID 生成器实例,这里会返回新建的或已经定义的实例 - 通过反射将 ID 生成器实例写入 bean
1 | java复制代码public class IdGeneratorBeanPostProcessor implements BeanPostProcessor { |
实现BeanPostProcessor
接口需要完成postProcessBeforeInitialization
和postProcessAfterInitialization
两个方法的定义。下图是 Spring 中 Bean 的实例化过程:
Spring 中 Bean 的实例化过程图示
从图中可以知道,Spring 调用BeanPostProcessor
的这两个方法时,bean 已经被实例化,所有能注入的属性都已经被注入了,是一个完整的 bean。而且两个方法的返回值,可以是原来的 bean 实例,也可以是包装后的实例,这就要看我们的定义了。
测试我们的代码
写一个测试用例,验证我们的实现是否生效:
1 | java复制代码@SpringBootTest |
运行结果为:
1 | ini复制代码DEFAULT => 1 |
可以看到,默认的 ID 生成器与定义名称为 group1 的 ID 生成器是分别生成的,符合预期。
文末思考
我们实现了通过BeanPostProcessor
实现自动注入自定义的业务对象,上面的实现还比较简单,有很多可以扩展的地方,比如工厂方法实现,可以借助 SPI 的方式更加灵活的创建 ID 生成器对象。同时,考虑到分布式场景,我们还可以在 ID 生成器实现类中,通过注入 rpc 实例,实现远程 ID 生成逻辑。
玩法无限,就看我们的想象了。可以关注公众号「看山的小屋」,回复“spring”获取源码。
推荐阅读
- SpringBoot 实战:一招实现结果的优雅响应
- SpringBoot 实战:如何优雅的处理异常
- SpringBoot 实战:通过 BeanPostProcessor 动态注入 ID 生成器
- SpringBoot 实战:自定义 Filter 优雅获取请求参数和响应结果
- SpringBoot 实战:优雅的使用枚举参数
- SpringBoot 实战:优雅的使用枚举参数(原理篇)
- SpringBoot 实战:在 RequestBody 中优雅的使用枚举参数
- SpringBoot 实战:在 RequestBody 中优雅的使用枚举参数(原理篇)
- SpringBoot 实战:JUnit5+MockMvc+Mockito 做好单元测试
- SpringBoot 实战:加载和读取资源文件内容
你好,我是看山。游于码界,戏享人生。如果文章对您有帮助,请点赞、收藏、关注。欢迎关注公众号「看山的小屋」,发现不一样的世界。
本文转载自: 掘金