我正在使用Spring-Retry进行一些数据库操作。在SQLRecoverableException上我重试三次(这假设如果失败三次,任何导致异常的东西都不是暂时的),在SQLTransientException上我无限期地重试(程序在不访问数据库的情况下不能做任何事情,所以它也可以一直重试,直到用户决定重新启动服务器),而在任何其他异常情况下我都不重试。我使用指数退避策略,基本重试时间为100毫秒,最大重试时间为30,000毫秒。
private static final int MAX_RECOVERABLE_RETRIES = 3;
private static final long INITIAL_INTERVAL = 100;
private static final long MAX_INTERVAL = 30 * 1000;
private static final double MULTIPLIER = 2.0;
public static RetryTemplate databaseTemplate() {
RetryTemplate template = new RetryTemplate();
ExceptionClassifierRetryPolicy retryPolicy = new ExceptionClassifierRetryPolicy();
Map<Class<? extends Throwable>, RetryPolicy> policyMap = new HashMap<>();
NeverRetryPolicy baseException = new NeverRetryPolicy();
SimpleRetryPolicy recoverablePolicy = new SimpleRetryPolicy();
recoverablePolicy.setMaxAttempts(MAX_RECOVERABLE_RETRIES);
AlwaysRetryPolicy transientPolicy = new AlwaysRetryPolicy();
policyMap.put(Exception.class, baseException);
policyMap.put(SQLRecoverableException.class, recoverablePolicy);
policyMap.put(SQLTransientException.class, transientPolicy);
retryPolicy.setPolicyMap(policyMap);
template.setRetryPolicy(retryPolicy);
ExponentialBackOffPolicy backOffPolicy = new ExponentialBackOffPolicy();
backOffPolicy.setInitialInterval(INITIAL_INTERVAL);
backOffPolicy.setMaxInterval(MAX_INTERVAL);
backOffPolicy.setMultiplier(MULTIPLIER);
template.setBackOffPolicy(backOffPolicy);
return template;
}理想情况下,我希望对所有SQLRecoverableExceptions使用100ms的固定回退,并且只对SQLTransientExceptions应用指数回退策略。我可以通过嵌套重试来实现这一点,但这会大大增加代码的复杂性--如果没有其他选择,我更愿意简单地将指数回退应用于SQLRecoverableException和SQLTransientException异常。
有没有办法使用一个重试模板将不同的退避策略应用到不同的异常?
发布于 2015-09-01 15:05:04
实际上,ExceptionClassifierRetryPolicy是一条可走的路。不过,我没能让它和policyMap一起工作。
下面是我如何使用它的:
@Component("yourRetryPolicy")
public class YourRetryPolicy extends ExceptionClassifierRetryPolicy
{
@PostConstruct
public void init()
{
final SimpleRetryPolicy simpleRetryPolicy = new SimpleRetryPolicy();
simpleRetryPolicy.setMaxAttempts( 3 );
this.setExceptionClassifier( new Classifier<Throwable, RetryPolicy>()
{
@Override
public RetryPolicy classify( Throwable classifiable )
{
if ( classifiable instanceof YourException )
{
return new NeverRetryPolicy();
}
// etc...
return simpleRetryPolicy;
}
});
}
}然后,您只需在重试模板上设置它:
@Autowired
@Qualifier("yourRetryPolicy")
private YourRetryPolicy yourRetryPolicy;
//...
RetryTemplate retryTemplate = new RetryTemplate();
retryTemplate.setRetryPolicy( yourRetryPolicy );发布于 2019-03-01 03:44:41
可接受的答案只处理特定于异常的RetryPolicy实例。Spring没有为特定于异常的BackOffPolicy实例提供任何开箱即用的功能。幸运的是,它的实现很简单。
import org.springframework.classify.Classifier
import org.springframework.classify.ClassifierSupport
import org.springframework.classify.SubclassClassifier
import org.springframework.retry.RetryContext
import org.springframework.retry.backoff.BackOffContext
import org.springframework.retry.backoff.BackOffInterruptedException
import org.springframework.retry.backoff.BackOffPolicy
import org.springframework.retry.backoff.NoBackOffPolicy
class ExceptionClassifierBackoffPolicy implements BackOffPolicy {
private static class ExceptionClassifierBackoffContext implements BackOffContext, BackOffPolicy {
Classifier<Throwable, BackOffPolicy> exceptionClassifier
RetryContext retryContext
Map<BackOffPolicy, BackOffContext> policyContextMap = [:]
ExceptionClassifierBackoffContext(Classifier<Throwable, BackOffPolicy> exceptionClassifier, RetryContext retryContext) {
this.exceptionClassifier = exceptionClassifier
this.retryContext = retryContext
}
@Override
BackOffContext start(RetryContext context) {
return null
}
@Override
void backOff(BackOffContext backOffContext) throws BackOffInterruptedException {
def policy = exceptionClassifier.classify(retryContext.lastThrowable)
def policyContext = policyContextMap.get(policy)
if (!policyContext) {
policyContext = policy.start(retryContext)
policyContextMap.put(policy, policyContext)
}
policy.backOff(policyContext)
}
}
private Classifier<Throwable, BackOffPolicy> exceptionClassifier = new ClassifierSupport<Throwable, BackOffPolicy>(new NoBackOffPolicy());
void setPolicyMap(Map<Class<? extends Throwable>, BackOffPolicy> policyMap) {
exceptionClassifier = new SubclassClassifier<Throwable, BackOffPolicy>(policyMap, new NoBackOffPolicy());
}
@Override
BackOffContext start(RetryContext context) {
return new ExceptionClassifierBackoffContext(exceptionClassifier, context)
}
@Override
void backOff(BackOffContext backOffContext) throws BackOffInterruptedException {
def classifierBackOffContext = (ExceptionClassifierBackoffContext) backOffContext
classifierBackOffContext.backOff(backOffContext)
}
}然后只需:
BackOffPolicy backOffPolicy = new ExceptionClassifierBackoffPolicy()
def policyMap = [
(RuntimeException): new FixedBackOffPolicy(backOffPeriod: 1000),
(IOException) : new ExponentialRandomBackOffPolicy(initialInterval: 500, maxInterval: 360000, multiplier: 2)
] as Map<Class<? extends Throwable>, BackOffPolicy>
backOffPolicy.policyMap = backoffPolicyMap发布于 2021-04-28 02:52:33
有点晚了,但是当你需要的是根据异常的类型有不同的退避时间时,扩展FixedBackoffPolicy应该可以做到这一点,而且非常简单。大致是这样的:
首先,创建退避策略类,该策略类接收每个异常类型具有不同退避期限的映射:
public class MultipleExceptionsBackoffPolicy extends FixedBackoffPolicy
{
private Classifier<Throwable, Long> classifier;
public MultipleExceptionsBackoffPolicy (final Map<Class<? extends Throwable>, Long> throwableBackoffMap) {
classifier = new SubclassClassifier<>(throwableBackoffMap, 5_000L) // default is 5s
}
@Override
protected void doBackOff() exception BackOffInterruptedException {
final backoff = classifier.classify(RetrySynchronizationManager.getContext().getLastThrowable());
setBackOffPeriod(backoff);
super.doBackOff();
}
}然后,您必须创建一个自定义拦截器,例如:
@Bean
public Object myCustomInterceptor (){
var exBackoffMap = Map.of(
ExceptionTypeOne.class, 2_000L, // If ExceptionTypeOne happens, backoff period is 2s
ExceptionTypeTwo.class, 7_000L // and if ExceptionTypeTwo happens, the backoff period is 7s
)
return RetryInterceptorBuilder
.stateless()
.retryPolicy(new SimpleRetryPolicy(3)) // always 3 attempts no matter what
.backOffPolicy(new MultipleExceptionsBackoffPolicy(exBackoffMap))
.build();
}最后,您只需在可重试中配置自定义拦截器:
@Retryable(interceptor="myCustomInterceptor")
public @interface MyRetryable {}https://stackoverflow.com/questions/32324172
复制相似问题