resubmit 渐进式防重复提交框架简介

resubmit

resubmit[1] 是一款为 java 设计的渐进式防止重复提交框架。

推荐阅读:

面试官:你们的项目中是怎么做防止重复提交的?

resubmit 渐进式防重复提交框架简介

创作目的

有时候手动加防止重复提交很麻烦,每次手动编写不利于复用。

所以希望从从简到繁实现一个工具,便于平时使用。

特性

渐进式实现,可独立 spring 使用

基于注解+字节码,配置灵活

支持编程式的调用

支持注解式,完美整合 spring

支持整合 spring-boot

变更日志[2]

快速开始

maven 引入

    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 中获取

spring 整合使用

maven 引入

    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 注解说明

@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");    }}

整合 spring-boot

maven 引入

    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 的缓存更加合适

自定义缓存 cache

实现缓存

只需要实现 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);    }}

core 中指定使用

在非 spring 项目中,可以在引导类中指定我们定义的缓存。

ResubmitBs resubmitBs = ResubmitBs.newInstance()                .cache(new MyDefineCache());UserService service = ResubmitProxy.getProxy(new UserService(), resubmitBs);

其他使用方式保持不变。

spring 中指定使用

在 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 的缓存策略已实现,后续有时间进行讲解。

Redis-Config[3]

开源地址

为了便于大家学习使用,目前防重复提交框架已开源。

欢迎大家 fork+star,鼓励一下老马~

https://github.com/houbb/resubmit

References

[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 条评论) “”
   
验证码:

相关文章

推荐文章