当我们调用一个接口可能由于网络等原因造成第一次失败,再去尝试就成功了,这就是重试机制,spring支持重试机制,并且在Spring Cloud中可以与Hystaix结合使用,可以避免访问到已经不正常的实例。 但是切记非幂等情况下慎用重试
一 加入依赖
<dependency> <groupId>org.springframework.retry</groupId> <artifactId>spring-retry</artifactId> </dependency> <dependency> <groupId>org.aspectj</groupId> <artifactId>aspectjweaver</artifactId> </dependency>
二 在主类上加入 @EnableRetry 注解
@EnableRetry //开启重试机制
@EnableAutoConfiguration //开启自动配置
@SpringBootApplication
public class WjKekingApplication {
public static void main(String[] args) {
SpringApplication.run(WjKekingApplication.class, args);
}
}
三 demo
package com.wj.project.keking.myTest.service;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.retry.annotation.Backoff;
import org.springframework.retry.annotation.Recover;
import org.springframework.retry.annotation.Retryable;
import org.springframework.stereotype.Service;
import java.time.LocalTime;
@Service
public class RetryService {
private final static Logger logger = LoggerFactory.getLogger(RetryService.class); private final int totalNum = 100000; /** * @Retryable的参数说明: •value:抛出指定异常才会重试 * •include:和value一样,默认为空,当exclude也为空时,默认所以异常 * •exclude:指定不处理的异常 * •maxAttempts:最大重试次数,默认3次 * •backoff:重试等待策略,默认使用@Backoff,@Backoff的value默认为1000L,我们设置为2000L;multiplier(指定延迟倍数)默认为0,表示固定暂停1秒后进行重试,如果把multiplier设置为1.5,则第一次重试为2秒,第二次为3秒,第三次为4.5秒。 */ @Retryable(value = Exception.class, maxAttempts = 3, backoff = @Backoff(delay = 2000L, multiplier = 1.5)) public int retry(int num) { logger.info("减库存开始" + LocalTime.now()); try { int i = 1 / 0; } catch (Exception e) { logger.error("illegal"); } if (num <= 0) { throw new IllegalArgumentException("数量不对"); } logger.info("减库存执行结束" + LocalTime.now()); return totalNum - num; } @Recover public void recover(Exception e) { logger.warn("减库存失败!!!" + LocalTime.now()); }
}
四 写一个简单的测试
import base.BaseTest;
import org.junit.Test;
import org.springframework.beans.factory.annotation.Autowired;
public class RetryServiceTest extends BaseTest {
@Autowired private RetryService retryService; @Test public void retry() { int count = retryService.retry(-1); System.out.println("库存为 :" + count); }
}
五 结果:
2018-07-28 01:12:09.836 INFO 31128 — [ main] o.s.b.w.embedded.tomcat.TomcatWebServer : Tomcat started on port(s): 55413 (http) with context path ‘’
2018-07-28 01:12:09.843 INFO 31128 — [ main] c.w.p.k.myTest.service.RetryServiceTest : Started RetryServiceTest in 7.178 seconds (JVM running for 10.657)
2018-07-28 01:12:10.584 INFO 31128 — [ main] c.w.p.k.myTest.service.RetryService : 减库存开始01:12:10.584
2018-07-28 01:12:10.584 ERROR 31128 — [ main] c.w.p.k.myTest.service.RetryService : illegal
2018-07-28 01:12:12.588 INFO 31128 — [ main] c.w.p.k.myTest.service.RetryService : 减库存开始01:12:12.588
2018-07-28 01:12:12.588 ERROR 31128 — [ main] c.w.p.k.myTest.service.RetryService : illegal
2018-07-28 01:12:15.588 INFO 31128 — [ main] c.w.p.k.myTest.service.RetryService : 减库存开始01:12:15.588
2018-07-28 01:12:15.588 ERROR 31128 — [ main] c.w.p.k.myTest.service.RetryService : illegal
2018-07-28 01:12:15.589 WARN 31128 — [ main] c.w.p.k.myTest.service.RetryService : 减库存失败!!!01:12:15.589
库存为 :100000
2018-07-28 01:12:15.604 INFO 31128 — [ Thread-3] ConfigServletWebServerApplicationContext : Closing org.springframework.boot.web.servlet.context.AnnotationConfigServletWebServerApplicationContext@1af2d44a: startup date [Sat Jul 28 01:12:03 CST 2018]; root of context hierarchy
2018-07-28 01:12:15.608 INFO 31128 — [ Thread-3] o.s.s.concurrent.ThreadPoolTaskExecutor : Shutting down ExecutorService ‘taskExecutor’
’
六 注意事项:
1、使用了@Retryable的方法不能在本类被调用,不然重试机制不会生效。也就是要标记为@Service,然后在其它类使用@Autowired注入或者@Bean去实例才能生效。
2 、要触发@Recover方法,那么在@Retryable方法上不能有返回值,只能是void才能生效。
3 、非幂等情况下慎用
4 、使用了@Retryable的方法里面不能使用try…catch包裹,要在方法上抛出异常,不然不会触发。