resubmit[1] 是一款为 java 设计的渐进式防止重复提交框架。
推荐阅读:
面试官:你们的项目中是怎么做防止重复提交的?
resubmit 渐进式防重复提交框架简介
有时候手动加防止重复提交很麻烦,每次手动编写不利于复用。
所以希望从从简到繁实现一个工具,便于平时使用。
•
渐进式实现,可独立 spring 使用
•
基于注解+字节码,配置灵活
•
支持编程式的调用
•
支持注解式,完美整合 spring
•
支持整合 spring-boot
变更日志[2]
com.github.houbb resubmit-core 1.0.0 •UserService.java
@Resubmit 对应的属性如下:
属性 | 说明 | 默认值 |
value() | 多久内禁止重复提交,单位为毫秒。 | 60000 |
@Resubmit(5000)public void queryInfo(final String id) { System.out.println("query info: " + id);}•测试代码
如果在指定时间差内,重复请求,则会抛出异常 ResubmitException
@Test(expected = ResubmitException.class)public void errorTest() { UserService service = ResubmitProxy.getProxy(new UserService()); service.queryInfo("1"); service.queryInfo("1");}相同的参数直接提交2次,就会报错。
•测试场景2
如果等待超过指定的 5s,就不会报错。
@Testpublic void untilTtlTest() { UserService service = ResubmitProxy.getProxy(new UserService()); service.queryInfo("1"); DateUtil.sleep(TimeUnit.SECONDS, 6); service.queryInfo("1");}ResubmitProxy.getProxy(new UserService()); 可以获取 UserService 对应的代理。
等价于:
ResubmitBs resubmitBs = ResubmitBs.newInstance() .cache(new CommonCacheServiceMap()) .keyGenerator(new KeyGenerator()) .tokenGenerator(new HttpServletRequestTokenGenerator());UserService service = ResubmitProxy.getProxy(new UserService(), resubmitBs);其中 ResubmitBs 作为引导类,对应的策略都支持自定义。
属性 | 说明 | 默认值 |
cache() | 缓存实现策略 | 默认为基于 ConcurrentHashMap 实现的基于内存的缓存实现 |
keyGenerator() | key 实现策略,用于唯一标识一个方法+参数,判断是否为相同的提交 | md5 策略 |
tokenGenerator() | token 实现策略,用于唯一标识一个用户。 | 从 HttpServletRequest 中的 header 属性 resubmit_token 中获取 |
com.github.houbb resubmit-spring 1.0.0 •UserService.java
@Servicepublic class UserService { @Resubmit(5000) public void queryInfo(final String id) { System.out.println("query info: " + id); }}•SpringConfig.java
@ComponentScan("com.github.houbb.resubmit.test.service")@EnableResubmit@Configurationpublic class SpringConfig {}@EnableResubmit 中用户可以指定对应的实现策略,便于更加灵活的适应业务场景。
和 ResubmitBs 中支持自定义的属性一一对应。
属性 | 说明 | 默认值 |
cache() | 缓存实现策略 | 默认为基于 ConcurrentHashMap 实现的基于内存的缓存实现 |
keyGenerator() | key 实现策略,用于唯一标识一个方法+参数,判断是否为相同的提交 | md5 策略 |
tokenGenerator() | token 实现策略,用于唯一标识一个用户。 | 从 HttpServletRequest 中的 header 属性 resubmit_token 中获取 |
@ContextConfiguration(classes = SpringConfig.class)@RunWith(SpringJUnit4ClassRunner.class)public class ResubmitSpringTest { @Autowired private UserService service; @Test(expected = ResubmitException.class) public void queryTest() { service.queryInfo("1"); service.queryInfo("1"); }} com.github.houbb resubmit-springboot-starter 1.0.0 •UserService.java
这个方法实现和前面的一样。
@Servicepublic class UserService { @Resubmit(5000) public void queryInfo(final String id) { System.out.println("query info: " + id); }}•Application.java
启动入口
@SpringBootApplicationpublic class ResubmitApplication { public static void main(String[] args) { SpringApplication.run(ResubmitApplication.class, args); }}@ContextConfiguration(classes = ResubmitApplication.class)@RunWith(SpringJUnit4ClassRunner.class)public class ResubmitSpringBootStarterTest { @Autowired private UserService service; @Test(expected = ResubmitException.class) public void queryTest() { service.queryInfo("1"); service.queryInfo("1"); }}上面提到 @EnableResubmit 中的策略支持自定义。
此处仅以 cache 为例,为了简单,默认是基于本地内存的缓存实现。
如果你不是单点应用,那么基于 redis 的缓存更加合适
只需要实现 ICommonCacheService 接口即可。
public class MyDefineCache extends CommonCacheServiceMap { // 这里只是作为演示,实际生产建议使用 redis 作为统一缓存 @Override public synchronized void set(String key, String value, long expireMills) { System.out.println("------------- 自定义的设置实现"); super.set(key, value, expireMills); }}在非 spring 项目中,可以在引导类中指定我们定义的缓存。
ResubmitBs resubmitBs = ResubmitBs.newInstance() .cache(new MyDefineCache());UserService service = ResubmitProxy.getProxy(new UserService(), resubmitBs);其他使用方式保持不变。
在 spring 项目中,我们需要调整一下配置,其他不变。
@ComponentScan("com.github.houbb.resubmit.test.service")@Configuration@EnableResubmit(cache = "myDefineCache")public class SpringDefineConfig { @Bean("myDefineCache") public ICommonCacheService myDefineCache() { return new MyDefineCache(); }}@EnableResubmit(cache = "myDefineCache") 指定我们自定义的缓存策略名称。
为了便于复用,基于 redis 的缓存策略已实现,后续有时间进行讲解。
Redis-Config[3]
为了便于大家学习使用,目前防重复提交框架已开源。
欢迎大家 fork+star,鼓励一下老马~
https://github.com/houbb/resubmit
[1] resubmit: https://github.com/houbb/resubmit
[2] 变更日志: https://github.com/houbb/resubmit/blob/master/CHANGELOG.md
[3] Redis-Config: https://github.com/houbb/redis-config
| 留言与评论(共有 0 条评论) “” |