如何全局捕捉 eureka 通过 Feign 远程调用出现的异常

万事皆有因

玩过 springcloud 的微服务架构的小伙伴们都知道,在我们通过 eureka 远程调用其他的微服务的时候,防止网络抖动或者程序异常等情况,往往会加上 hystrix 断路器对异常进行处理,但是往往在处理的时候一般都是记录一些日志告诉开发者哪里出问题了,参数是什么,又或者其他的处理

然而,很多人使用 Feign 的时候 rollback 都是继承被调用的 FeignClient 的一个子类,然后里面重写接口的方法,记录日志等等操作。

想想一下,其实处理这些 rpc 异常的方式基本差不多,然后每一个方法都需要重新写一遍,强迫症严重的人肯定不允许这样的事情发生,那么我们该如何做到通过一个方法将这些错误日志将调不同接口方法异常打印出来呢?

突然灵光一闪

该怎么做呢?当时我想啊,若能动态生成其子类的方法是不是可以达到目的呢?于是通过动态代理去实现,下面给出代码。

  • rollback 工厂类

import com.qts.agent.web.util.ReflectionUtil;
import feign.hystrix.FallbackFactory;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.stereotype.Component;

import java.lang.reflect.Proxy;

/**
 * 基础的远程异常捕捉
 *
 * @author 阿导
 * @CopyRight 万物皆导
 * @created 2018年08月09日 14:56:00
 * @Modified_By 阿导 2018/8/9 14:56
 */
@Component
public class BaseFailedFactory<T> implements FallbackFactory<T> {
  /**
   * 没别的意思,就想记一个日志
   */
  private Logger logger = LoggerFactory.getLogger(this.getClass());
  
  
  @Override
  public T create(Throwable throwable) {
    T newProxyInstance = (T) Proxy.newProxyInstance(
        getEntityClass().getClassLoader(),
        new Class[] { getEntityClass() }, new MethodProxy(logger));
    return newProxyInstance;
  }
  
  /**
   * 获取需要操作的实体类class
   *
   * @return
   */
  private <T> Class<? extends T> getEntityClass() {
    return ReflectionUtil.getSuperClassGenricType(getClass());
  }
}



  • 代理方法类
import com.alibaba.fastjson.JSON;
import org.slf4j.Logger;

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;

/**
 * 方法代理
 *
 * @author 阿导
 * @CopyRight 万物皆导
 * @created 2018/8/9 15:43
 * @Modified_By 阿导 2018/8/9 15:43
 */
public class MethodProxy implements InvocationHandler {
  private Logger logger;
  
  public MethodProxy(Logger logger) {
    this.logger = logger;
  }
  
  @Override
  public Object invoke(Object proxy, Method method, Object[] args) {
    // 执行之前
    if (Object.class.equals(method.getDeclaringClass())) {
      try {
        return method.invoke(this, args);
      } catch (Throwable t) {
        t.printStackTrace();
      }
    } else {
      return run(method, args);
    }
    // 执行之后
    return null;
  }
  
  /**
   *
   *
   * @author 阿导
   * @time 2018/8/9 16:35
   * @CopyRight 万物皆导
   * @param method
   * @param args
   * @return
   */
  private Object run(Method method, Object[] args) {
    logger.error("远程调用失败,目标:" + method.getDeclaringClass());
    logger.error("参数:" + JSON.toJSONString(args));
    return null;
  } 
}


  • 反射工具类获取泛型的具体的类

点击前往源码

就那么轻轻一测试

测试就不用说了哈,要么是调用的服务宕机,要么使其异常,看是否能处理,大家明白了吗?若有不清楚的地方,麻烦再留言区留下金言玉语,当然写的不足之处,还望海涵。

需要云服务器的不要错过优惠

阿里云低价购买云服务,值得一看

你可能感兴趣的:(spring-boot,java)