需要在Spring中自动重新调用失败的操作吗?想了解更多请查看有关Spring Retry的本教程。
几天前,我注意到有一群人在询问如何使用Spring Retry。在我进入示例代码之前,让我快速解释Spring Retry背后的目的。Spring Retry提供了自动重新调用失败操作的功能。对瞬间错误(如瞬间网络故障)很有用。Spring Retry提供对流程和基于策略的行为的声明性控制,易于扩展和自定义。
你在这里可以找到完整的代码。
Maven Dependencies
<dependencies>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.retry</groupId>
<artifactId>spring-retry</artifactId>
</dependency>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-aop</artifactId>
</dependency>
<dependency>
<groupId>org.aspectj</groupId>
<artifactId>aspectjweaver</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-test</artifactId>
<scope>test</scope>
</dependency>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-test</artifactId>
<scope>test</scope>
</dependency>
</dependencies>
Enable Retry
package com.chrisshayan.example.springretry;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.retry.annotation.EnableRetry;
@EnableRetry
@SpringBootApplication
public class SpringRetryApplication {
public static void main(String[] args) {
SpringApplication.run(SpringRetryApplication.class, args);
}
}
Using Retry With Annotations
package com.chrisshayan.example.springretry;
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;
@Service
public class SampleRetryService {
private static final Logger LOGGER = LoggerFactory.getLogger(SampleRetryService.class);
private static int COUNTER = 0;
@Retryable(
value = {TypeOneException.class, TypeTwoException.class},
maxAttempts = 4, backoff = @Backoff(2000))
public String retryWhenException() throws TypeOneException, TypeTwoException {
COUNTER++;
LOGGER.info("COUNTER = " + COUNTER);
if(COUNTER == 1)
throw new TypeOneException();
else if(COUNTER == 2)
throw new TypeTwoException();
else
throw new RuntimeException();
}
@Recover
public String recover(Throwable t) {
LOGGER.info("SampleRetryService.recover");
return "Error Class :: " + t.getClass().getName();
}
}
为了使您的测试类起作用,重试需要在适当的上下文中。这是因为我们需要另一个包含重试的服务。以下就是:
package com.chrisshayan.example.springretry;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
@Service
public class SampleRetryClientService {
@Autowired
private SampleRetryService sampleRetryService;
public String callRetryService() throws TypeOneException, TypeTwoException {
return sampleRetryService.retryWhenException();
}
}
Test Class
package com.chrisshayan.example.springretry;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;
@RunWith(SpringJUnit4ClassRunner.class)
@SpringBootTest
public class SpringRetryApplicationTests {
private static final Logger LOGGER = LoggerFactory.getLogger(SpringRetryApplicationTests.class);
@Autowired
private SampleRetryClientService client;
@Test
public void contextLoads() {
}
@Test
public void sampleRetryService() {
try {
final String message = client.callRetryService();
LOGGER.info("message = " + message);
} catch (TypeOneException | TypeTwoException e) {
e.printStackTrace();
}
}
}
console
2018-07-10 23:42:45.528 INFO 14583 --- [ main] c.c.e.springretry.SampleRetryService : COUNTER = 1
2018-07-10 23:42:47.534 INFO 14583 --- [ main] c.c.e.springretry.SampleRetryService : COUNTER = 2
2018-07-10 23:42:49.538 INFO 14583 --- [ main] c.c.e.springretry.SampleRetryService : COUNTER = 3
2018-07-10 23:42:49.539 INFO 14583 --- [ main] c.c.e.springretry.SampleRetryService : SampleRetryService.recover
2018-07-10 23:42:49.539 INFO 14583 --- [ main] c.c.e.s.SpringRetryApplicationTests : message = Error Class :: java.lang.RuntimeException
Spring Retry中有更多功能,例如无状态重试,状态重试以及不同的重试策略和侦听器。你可以在这里阅读更多。