目錄
重試的使用場景
在很多業務場景中,為了排除系統中的各種不穩定因素,以及邏輯上的錯誤,並最大概率保證獲得預期的結果,重試機制都是必不可少的。
尤其是呼叫遠端服務,在高併發場景下,很可能因為伺服器響應延遲或者網路原因,造成我們得不到想要的結果,或者根本得不到響應。這個時候,一個優雅的重試呼叫機制,可以讓我們更大概率保證得到預期的響應。
通常情況下,我們會通過定時任務進行重試。例如某次操作失敗,則記錄下來,當定時任務再次啟動,則將資料放到定時任務的方法中,重新跑一遍。最終直至得到想要的結果為止。
無論是基於定時任務的重試機制,還是我們自己寫的簡單的重試器,缺點都是重試的機制太單一,而且實現起來不優雅。
如何優雅地設計重試實現
一個完備的重試實現,要很好地解決如下問題:
- 什麼條件下重試
- 什麼條件下停止
- 如何停止重試
- 停止重試等待多久
- 如何等待
- 請求時間限制
- 如何結束
- 如何監聽整個重試過程
並且,為了更好地封裝性,重試的實現一般分為兩步:
- 使用工廠模式構造重試器
- 執行重試方法並得到結果
一個完整的重試流程可以簡單示意為:
guava-retrying基礎用法
guava-retrying是基於谷歌的核心類庫guava的重試機制實現,可以說是一個重試利器。
下面就快速看一下它的用法。
1.Maven配置
<!-- https://mvnrepository.com/artifact/com.github.rholder/guava-retrying -->
<dependency>
<groupId>com.github.rholder</groupId>
<artifactId>guava-retrying</artifactId>
<version>2.0.0</version>
</dependency>
複製程式碼
需要注意的是,此版本依賴的是27.0.1版本的guava。如果你專案中的guava低幾個版本沒問題,但是低太多就不相容了。這個時候你需要升級你專案的guava版本,或者直接去掉你自己的guava依賴,使用guava-retrying傳遞過來的guava依賴。
2.實現Callable
Callable<Boolean> callable = new Callable<Boolean>() {
public Boolean call() throws Exception {
return true; // do something useful here
}
};
複製程式碼
Callable的call方法中是你自己實際的業務呼叫。
- 通過RetryerBuilder構造Retryer
Retryer<Boolean> retryer = RetryerBuilder.<Boolean>newBuilder()
.retryIfResult(Predicates.<Boolean>isNull())
.retryIfExceptionOfType(IOException.class)
.retryIfRuntimeException()
.withStopStrategy(StopStrategies.stopAfterAttempt(3))
.build();
複製程式碼
- 使用重試器執行你的業務
retryer.call(callable);
複製程式碼
下面是完整的參考實現。
public Boolean test() throws Exception {
//定義重試機制
Retryer<Boolean> retryer = RetryerBuilder.<Boolean>newBuilder()
//retryIf 重試條件
.retryIfException()
.retryIfRuntimeException()
.retryIfExceptionOfType(Exception.class)
.retryIfException(Predicates.equalTo(new Exception()))
.retryIfResult(Predicates.equalTo(false))
//等待策略:每次請求間隔1s
.withWaitStrategy(WaitStrategies.fixedWait(1, TimeUnit.SECONDS))
//停止策略 : 嘗試請求6次
.withStopStrategy(StopStrategies.stopAfterAttempt(6))
//時間限制 : 某次請求不得超過2s , 類似: TimeLimiter timeLimiter = new SimpleTimeLimiter();
.withAttemptTimeLimiter(AttemptTimeLimiters.fixedTimeLimit(2, TimeUnit.SECONDS))
.build();
//定義請求實現
Callable<Boolean> callable = new Callable<Boolean>() {
int times = 1;
@Override
public Boolean call() throws Exception {
log.info("call times={}", times);
times++;
if (times == 2) {
throw new NullPointerException();
} else if (times == 3) {
throw new Exception();
} else if (times == 4) {
throw new RuntimeException();
} else if (times == 5) {
return false;
} else {
return true;
}
}
};
//利用重試器呼叫請求
return retryer.call(callable);
}
複製程式碼
guava-retrying實現原理
guava-retrying的核心是Attempt類、Retryer類以及一些Strategy(策略)相關的類。
- Attempt
Attempt既是一次重試請求(call),也是請求的結果,並記錄了當前請求的次數、是否包含異常和請求的返回值。
/**
* An attempt of a call, which resulted either in a result returned by the call,
* or in a Throwable thrown by the call.
*
* @param <V> The type returned by the wrapped callable.
* @author JB
*/
public interface Attempt<V>
複製程式碼
- Retryer
Retryer通過RetryerBuilder這個工廠類進行構造。RetryerBuilder負責將定義的重試策略賦值到Retryer物件中。
在Retryer執行call方法的時候,會將這些重試策略一一使用。
下面就看一下Retryer的call方法的具體實現。
/**
* Executes the given callable. If the rejection predicate
* accepts the attempt, the stop strategy is used to decide if a new attempt
* must be made. Then the wait strategy is used to decide how much time to sleep
* and a new attempt is made.
*
* @param callable the callable task to be executed
* @return the computed result of the given callable
* @throws ExecutionException if the given callable throws an exception, and the
* rejection predicate considers the attempt as successful. The original exception
* is wrapped into an ExecutionException.
* @throws RetryException if all the attempts failed before the stop strategy decided
* to abort, or the thread was interrupted. Note that if the thread is interrupted,
* this exception is thrown and the thread's interrupt status is set.
*/
public V call(Callable<V> callable) throws ExecutionException, RetryException {
long startTime = System.nanoTime();
//說明: 根據attemptNumber進行迴圈——也就是重試多少次
for (int attemptNumber = 1; ; attemptNumber++) {
//說明:進入方法不等待,立即執行一次
Attempt<V> attempt;
try {
//說明:執行callable中的具體業務
//attemptTimeLimiter限制了每次嘗試等待的時常
V result = attemptTimeLimiter.call(callable);
//利用呼叫結果構造新的attempt
attempt = new ResultAttempt<V>(result, attemptNumber, TimeUnit.NANOSECONDS.toMillis(System.nanoTime() - startTime));
} catch (Throwable t) {
attempt = new ExceptionAttempt<V>(t, attemptNumber, TimeUnit.NANOSECONDS.toMillis(System.nanoTime() - startTime));
}
//說明:遍歷自定義的監聽器
for (RetryListener listener : listeners) {
listener.onRetry(attempt);
}
//說明:判斷是否滿足重試條件,來決定是否繼續等待並進行重試
if (!rejectionPredicate.apply(attempt)) {
return attempt.get();
}
//說明:此時滿足停止策略,因為還沒有得到想要的結果,因此丟擲異常
if (stopStrategy.shouldStop(attempt)) {
throw new RetryException(attemptNumber, attempt);
} else {
//說明:執行預設的停止策略——執行緒休眠
long sleepTime = waitStrategy.computeSleepTime(attempt);
try {
//說明:也可以執行定義的停止策略
blockStrategy.block(sleepTime);
} catch (InterruptedException e) {
Thread.currentThread().interrupt();
throw new RetryException(attemptNumber, attempt);
}
}
}
}
複製程式碼
Retryer執行過程如下。
guava-retrying高階用法
基於guava-retrying的實現原理,我們可以根據實際業務來確定自己的重試策略。
下面以資料同步
這種常規系統業務為例,自定義重試策略。
如下實現基於Spring Boot 2.1.2.RELEASE版本。
並使用Lombok簡化Bean。
<dependency>
<groupId>org.projectlombok</groupId>
<artifactId>lombok</artifactId>
<optional>true</optional>
</dependency>
複製程式碼
業務描述
當商品建立以後,需要另外設定商品的價格。由於兩個操作是有兩個人進行的,因此會出現如下問題,即商品沒有建立,但是價格資料卻已經建好了。遇到這種情況,價格資料需要等待商品正常建立以後,繼續完成同步。
我們通過一個http請求進行商品的建立,同時通過一個定時器來修改商品的價格。
當商品不存在,或者商品的數量小於1的時候,商品的價格不能設定。需要等商品成功建立且數量大於0的時候,才能將商品的價格設定成功。
實現過程
- 自定義重試阻塞策略
預設的阻塞策略是執行緒休眠,這裡使用自旋鎖實現,不阻塞執行緒。
package net.ijiangtao.tech.framework.spring.ispringboot.demo.retryer.guava.strategy;
import com.github.rholder.retry.BlockStrategy;
import lombok.NoArgsConstructor;
import lombok.extern.slf4j.Slf4j;
import java.time.Duration;
import java.time.LocalDateTime;
/**
* 自旋鎖的實現, 不響應執行緒中斷
*/
@Slf4j
@NoArgsConstructor
public class SpinBlockStrategy implements BlockStrategy {
@Override
public void block(long sleepTime) throws InterruptedException {
LocalDateTime startTime = LocalDateTime.now();
long start = System.currentTimeMillis();
long end = start;
log.info("[SpinBlockStrategy]...begin wait.");
while (end - start <= sleepTime) {
end = System.currentTimeMillis();
}
//使用Java8新增的Duration計算時間間隔
Duration duration = Duration.between(startTime, LocalDateTime.now());
log.info("[SpinBlockStrategy]...end wait.duration={}", duration.toMillis());
}
}
複製程式碼
- 自定義重試監聽器
RetryListener可以監控多次重試過程,並可以使用attempt
做一些額外的事情。
package net.ijiangtao.tech.framework.spring.ispringboot.demo.retryer.guava.listener;
import com.github.rholder.retry.Attempt;
import com.github.rholder.retry.RetryListener;
import lombok.extern.slf4j.Slf4j;
@Slf4j
public class RetryLogListener implements RetryListener {
@Override
public <V> void onRetry(Attempt<V> attempt) {
// 第幾次重試,(注意:第一次重試其實是第一次呼叫)
log.info("retry time : [{}]", attempt.getAttemptNumber());
// 距離第一次重試的延遲
log.info("retry delay : [{}]", attempt.getDelaySinceFirstAttempt());
// 重試結果: 是異常終止, 還是正常返回
log.info("hasException={}", attempt.hasException());
log.info("hasResult={}", attempt.hasResult());
// 是什麼原因導致異常
if (attempt.hasException()) {
log.info("causeBy={}" , attempt.getExceptionCause().toString());
} else {
// 正常返回時的結果
log.info("result={}" , attempt.getResult());
}
log.info("log listen over.");
}
}
複製程式碼
- 自定義Exception
有些異常需要重試,有些不需要。
package net.ijiangtao.tech.framework.spring.ispringboot.demo.retryer.guava.exception;
/**
* 當丟擲這個異常的時候,表示需要重試
*/
public class NeedRetryException extends Exception {
public NeedRetryException(String message) {
super("NeedRetryException can retry."+message);
}
}
複製程式碼
- 實現具體重試業務與Callable介面
使用call方法呼叫自己的業務。
package net.ijiangtao.tech.framework.spring.ispringboot.demo.retryer.guava.model;
import lombok.AllArgsConstructor;
import lombok.Data;
import java.math.BigDecimal;
/**
* 商品model
*/
@Data
@AllArgsConstructor
public class Product {
private Long id;
private String name;
private Integer count;
private BigDecimal price;
}
複製程式碼
package net.ijiangtao.tech.framework.spring.ispringboot.demo.retryer.guava.repository;
import net.ijiangtao.tech.framework.spring.ispringboot.demo.retryer.guava.model.Product;
import org.springframework.stereotype.Repository;
import java.math.BigDecimal;
import java.util.ArrayList;
import java.util.List;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.atomic.AtomicLong;
/**
* 商品DAO
*/
@Repository
public class ProductRepository {
private static ConcurrentHashMap<Long,Product> products=new ConcurrentHashMap();
private static AtomicLong ids=new AtomicLong(0);
public List<Product> findAll(){
return new ArrayList<>(products.values());
}
public Product findById(Long id){
return products.get(id);
}
public Product updatePrice(Long id, BigDecimal price){
Product p=products.get(id);
if (null==p){
return p;
}
p.setPrice(price);
return p;
}
public Product addProduct(Product product){
Long id=ids.addAndGet(1);
product.setId(id);
products.put(id,product);
return product;
}
}
複製程式碼
package net.ijiangtao.tech.framework.spring.ispringboot.demo.retryer.guava.service;
import lombok.extern.slf4j.Slf4j;
import net.ijiangtao.tech.framework.spring.ispringboot.demo.retryer.guava.exception.NeedRetryException;
import net.ijiangtao.tech.framework.spring.ispringboot.demo.retryer.guava.model.Product;
import net.ijiangtao.tech.framework.spring.ispringboot.demo.retryer.guava.repository.ProductRepository;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Component;
import java.math.BigDecimal;
import java.util.HashMap;
import java.util.Map;
import java.util.concurrent.Callable;
/**
* 業務方法實現
*/
@Component
@Slf4j
public class ProductInformationHander implements Callable<Boolean> {
@Autowired
private ProductRepository pRepo;
private static Map<Long, BigDecimal> prices = new HashMap<>();
static {
prices.put(1L, new BigDecimal(100));
prices.put(2L, new BigDecimal(200));
prices.put(3L, new BigDecimal(300));
prices.put(4L, new BigDecimal(400));
prices.put(8L, new BigDecimal(800));
prices.put(9L, new BigDecimal(900));
}
@Override
public Boolean call() throws Exception {
log.info("sync price begin,prices size={}", prices.size());
for (Long id : prices.keySet()) {
Product product = pRepo.findById(id);
if (null == product) {
throw new NeedRetryException("can not find product by id=" + id);
}
if (null == product.getCount() || product.getCount() < 1) {
throw new NeedRetryException("product count is less than 1, id=" + id);
}
Product updatedP = pRepo.updatePrice(id, prices.get(id));
if (null == updatedP) {
return false;
}
prices.remove(id);
}
log.info("sync price over,prices size={}", prices.size());
return true;
}
}
複製程式碼
- 構造重試器Retryer
將上面的實現作為引數,構造Retryer。
package net.ijiangtao.tech.framework.spring.ispringboot.demo.retryer.guava.service;
import com.github.rholder.retry.*;
import net.ijiangtao.tech.framework.spring.ispringboot.demo.retryer.guava.exception.NeedRetryException;
import net.ijiangtao.tech.framework.spring.ispringboot.demo.retryer.guava.listener.RetryLogListener;
import net.ijiangtao.tech.framework.spring.ispringboot.demo.retryer.guava.strategy.SpinBlockStrategy;
import org.springframework.stereotype.Component;
import java.util.concurrent.TimeUnit;
/**
* 構造重試器
*/
@Component
public class ProductRetryerBuilder {
public Retryer build() {
//定義重試機制
Retryer<Boolean> retryer = RetryerBuilder.<Boolean>newBuilder()
//retryIf 重試條件
//.retryIfException()
//.retryIfRuntimeException()
//.retryIfExceptionOfType(Exception.class)
//.retryIfException(Predicates.equalTo(new Exception()))
//.retryIfResult(Predicates.equalTo(false))
.retryIfExceptionOfType(NeedRetryException.class)
//等待策略:每次請求間隔1s
.withWaitStrategy(WaitStrategies.fixedWait(1, TimeUnit.SECONDS))
//停止策略 : 嘗試請求3次
.withStopStrategy(StopStrategies.stopAfterAttempt(3))
//時間限制 : 某次請求不得超過2s , 類似: TimeLimiter timeLimiter = new SimpleTimeLimiter();
.withAttemptTimeLimiter(AttemptTimeLimiters.fixedTimeLimit(2, TimeUnit.SECONDS))
//預設的阻塞策略:執行緒睡眠
//.withBlockStrategy(BlockStrategies.threadSleepStrategy())
//自定義阻塞策略:自旋鎖
.withBlockStrategy(new SpinBlockStrategy())
//自定義重試監聽器
.withRetryListener(new RetryLogListener())
.build();
return retryer;
}
}
複製程式碼
- 與定時任務結合執行Retryer
定時任務只需要跑一次,但是實際上實現了所有的重試策略。這樣大大簡化了定時器的設計。
首先使用@EnableScheduling
宣告專案支援定時器註解。
@SpringBootApplication
@EnableScheduling
public class DemoRetryerApplication {
public static void main(String[] args) {
SpringApplication.run(DemoRetryerApplication.class, args);
}
}
複製程式碼
package net.ijiangtao.tech.framework.spring.ispringboot.demo.retryer.guava.task;
import com.github.rholder.retry.Retryer;
import net.ijiangtao.tech.framework.spring.ispringboot.demo.retryer.guava.service.ProductInformationHander;
import net.ijiangtao.tech.framework.spring.ispringboot.demo.retryer.guava.service.ProductRetryerBuilder;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.scheduling.annotation.Scheduled;
import org.springframework.stereotype.Component;
/**
* 商品資訊定時器
*/
@Component
public class ProductScheduledTasks {
@Autowired
private ProductRetryerBuilder builder;
@Autowired
private ProductInformationHander hander;
/**
* 同步商品價格定時任務
* @Scheduled(fixedDelay = 30000) :上一次執行完畢時間點之後30秒再執行
*/
@Scheduled(fixedDelay = 30*1000)
public void syncPrice() throws Exception{
Retryer retryer=builder.build();
retryer.call(hander);
}
}
複製程式碼
執行結果:由於並沒有商品,因此重試以後,丟擲異常。
2019-二月-28 14:37:52.667 INFO [scheduling-1] n.i.t.f.s.i.d.r.g.l.RetryLogListener - log listen over.
2019-二月-28 14:37:52.672 ERROR [scheduling-1] o.s.s.s.TaskUtils$LoggingErrorHandler - Unexpected error occurred in scheduled task.
com.github.rholder.retry.RetryException: Retrying failed to complete successfully after 3 attempts.
at com.github.rholder.retry.Retryer.call(Retryer.java:174)
複製程式碼
你也可以增加一些商品資料,看一下重試成功的效果。
完整示例程式碼在這裡。
使用中遇到的問題
Guava版本衝突
由於專案中依賴的guava版本過低,啟動專案時出現瞭如下異常。
java.lang.NoSuchMethodError: com.google.common.util.concurrent.MoreExecutors.sameThreadExecutor()Lcom/google/common/util/concurrent/ListeningExecutorService;
at org.apache.curator.framework.listen.ListenerContainer.addListener(ListenerContainer.java:41)
at com.bzn.curator.ZkOperator.getZkClient(ZkOperator.java:207)
at com.bzn.curator.ZkOperator.checkExists(ZkOperator.java:346)
at com.bzn.curator.watcher.AbstractWatcher.initListen(AbstractWatcher.java:87)
at com.bzn.web.listener.NebulaSystemInitListener.initZkWatcher(NebulaSystemInitListener.java:84)
at com.bzn.web.listener.NebulaSystemInitListener.contextInitialized(NebulaSystemInitListener.java:33)
at org.apache.catalina.core.StandardContext.listenerStart(StandardContext.java:4939)
at org.apache.catalina.core.StandardContext.startInternal(StandardContext.java:5434)
at org.apache.catalina.util.LifecycleBase.start(LifecycleBase.java:150)
at org.apache.catalina.core.ContainerBase$StartChild.call(ContainerBase.java:1559)
at org.apache.catalina.core.ContainerBase$StartChild.call(ContainerBase.java:1549)
at java.util.concurrent.FutureTask.run(FutureTask.java:266)
at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1149)
at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:624)
at java.lang.Thread.run(Thread.java:748)
複製程式碼
因此,要排除專案中低版本的guava依賴。
<exclusion>
<groupId>com.google.guava</groupId>
<artifactId>guava</artifactId>
</exclusion>
複製程式碼
同時,由於Guava在新版本中移除了sameThreadExecutor
方法,但目前專案中的ZK需要此方法,因此需要手動設定合適的guava版本。
果然,在19.0版本中MoreExecutors的此方法依然存在,只是標註為過期了。
@Deprecated
@GwtIncompatible("TODO")
public static ListeningExecutorService sameThreadExecutor() {
return new DirectExecutorService();
}
複製程式碼
宣告依賴的guava版本改為19.0即可。
<!-- https://mvnrepository.com/artifact/com.google.guava/guava -->
<dependency>
<groupId>com.google.guava</groupId>
<artifactId>guava</artifactId>
<version>19.0</version>
</dependency>
複製程式碼
動態調節重試策略
在實際使用過程中,有時經常需要調整重試的次數、等待的時間等重試策略,因此,將重試策略的配置引數化儲存,可以動態調節。
例如在秒殺、雙十一購物節等時期增加等待的時間與重試次數,以保證錯峰請求。在平時,可以適當減少等待時間和重試次數。
對於系統關鍵性業務,如果多次重試步成功,可以通過RetryListener進行監控與報警。
links:
- https://github.com/rholder/guava-retrying
- https://github.com/spring-projects/spring-retry
- https://zhuanlan.zhihu.com/p/57970758
- https://blog.csdn.net/sinat_26342009/article/details/88045701
- https://github.com/ispringboot/demos/tree/release-retryer-v1/demo-retryer
author: ijiangtao