前言

微信支付成功之后,会对商户系统发送异步回调请求,来通知商户支付成功。

需要注意的是:

  • 同样的通知可能会多次发送给商户系统,商户系统必须能够正确处理重复的通知
  • 后台通知交互时,如果微信收到商户的应答不符合规范或超时,微信会判定本次通知失败,重新发送通知,直到成功为止
  • 在订单状态不明或者没有收到微信支付结果通知的情况下,建议商户主动调用微信支付【查询订单API】确认订单状态。

集成

在调用外部服务进行操作时,常常因为网络抖动、服务方进行限流等因素造成查询失败。为了克服这些问题,引入了重试机制。

在通知一直不成功的情况下,微信总共会发起多次通知,通知频率为15s/15s/30s/3m/10m/20m/30m/30m/30m/60m/3h/3h/3h/6h/6h - 总计 24h4m),但微信不保证通知最终一定能成功

重试原则

  • 查询可以进行重试
  • 写操作要慎重,除非业务方支持重入

配置文件 pom.xml 引入:

<dependency>
    <groupId>org.springframework.retry</groupId>
    <artifactId>spring-retry</artifactId>
    <version>1.2.5.RELEASE</version>
</dependency>

启动类新增注解 @EnableRetry:

@EnableRetry
@EnableAsync
@EnableCaching
@EnableScheduling
@SpringBootApplication
public class Application extends SpringBootServletInitializer {

    private static final Logger logger = LoggerFactory.getLogger(Application.class);

    public static void main(String[] args) {
        SpringApplication.run(Application.class, args);
        logger.info("PayCloud 支付分账系统");
    }

    @Override
    protected SpringApplicationBuilder configure(SpringApplicationBuilder application) {
        return application.sources(Application.class);
    }
}

回调商家客户端:

@Async
@Override
@Transactional(rollbackFor=Exception.class)
@Retryable(value= {Exception.class},maxAttempts = 10,backoff = @Backoff(delay = 2000L,multiplier = 2))
public void notifyApp(Map<String,Object> params) throws Exception {
    Date date = new Date();
    SimpleDateFormat format = new SimpleDateFormat("HH:mm:ss");
    String str = format.format(date);
    System.out.println("现在时间:" + str);
    /**
     * 处理业务逻辑
     */
    throw new Exception("应用回调异常");
}

打印时间如下:

现在时间:21:36:40
现在时间:21:36:42
现在时间:21:36:46
现在时间:21:36:54
现在时间:21:37:10
现在时间:21:37:40
现在时间:21:38:10
现在时间:21:38:40
现在时间:21:39:10
现在时间:21:39:40

需要注意的是,最后几次重试间隔定格在了30s,原因是参数中有个maxdelay属性,默认是30s。时间间隔是取{delay,maxDelay}的最小值。如果想继续递增执行,需要将 maxDelay 设置为理想的数值。

说明

参数 @Retryable 说明

  • value:抛出指定异常才会重试
  • include:和value一样,默认为空,当exclude也为空时,默认所以异常
  • exclude:指定不处理的异常
  • maxAttempts:最大重试次数,默认3次
  • backoff:重试等待策略,默认使用@Backoff,@Backoff的value默认为1000L,我们设置为2000L;multiplier(指定延迟倍数)默认为0,表示固定暂停1秒后进行重试,如果把multiplier设置为2,则第一次重试为2秒,第二次为4秒,第三次为8秒。

小结

总的来说,这玩意还是比较不错的,但是如果遇到程序崩溃或者集群扩展那问题就来了,显然是不能满足我们实际的业务需求。后期可以使用高可用的延迟队列或者手动补偿的方式来解决!