美文网首页
@ControllerAdvice 不生效踩坑

@ControllerAdvice 不生效踩坑

作者: 得鹿梦为鱼 | 来源:发表于2020-08-18 00:13 被阅读0次

背景

遇到一个bug,同一个接口,在开发环境和其他环境,期望得到的异常请求结果居然不同,但每个环境的代码都是一致的

排查

通过走读代码 + debug,发现我们系统自定义的 异常处理类,除开发外,其余环境没有生效!!!

自定义异常处理类示例

为什么没有生效?

原因可能有以下几种:
1.没被Spring加载
没被加载又有几种因素:比如没被扫描到,初始化bean的时候报错了等
-- 这种情况不存在,因为开发环境是ok

2.还有其他类似被标注了 @ControllerAdvice 的类存在,由于类加载顺序问题,被其他 ControllerAdvice 优先执行了

-- 然而找遍工程,没有


image.png

现在只能怀疑是不是在依赖的jar中存在类似的处理

下一个问题,怎么从jar里找这个注解是否被使用了呢?

百度了下,貌似没有工具能做到从class里查数据的

利用ClassLoader加载指定包下面所有的Class对象

刚好我们的工程在启动类那里,指定了扫描的包,那么如果存在类似功能的Class,并且要生效,那么一定是能被扫描到的

@SpringBootApplication(scanBasePackages = {"com.xxx"})

去网上去找了一份代码,获取到了指定包下所有的Class对象,同时也输出目标结果

扫描指定路径下并返回拥有指定注解的class

通过这份代码输出的结果,终于定位到了问题

这份从网上引用的代码放到文章最后,仅供参考,不做其他任何非法使用

解决

对自己系统的异常处理类添加优先级

@Priority(1)

示例:


添加@Priority

网上摘抄加载Class代码示例


public class Scanner {

    /**
     * 从包package中获取所有的Class
     *
     * @param packageName
     * @return
     */
    public Set<Class<?>> getClasses(String packageName) throws Exception {

        // 第一个class类的集合
        //List<Class<?>> classes = new ArrayList<Class<?>>();
        Set<Class<?>> classes = new HashSet<>();
        // 是否循环迭代
        boolean recursive = true;
        // 获取包的名字 并进行替换
        String packageDirName = packageName.replace('.', '/');
        // 定义一个枚举的集合 并进行循环来处理这个目录下的things
        Enumeration<URL> dirs;
        try {
            dirs = Thread.currentThread().getContextClassLoader().getResources(packageDirName);
            // 循环迭代下去
            while (dirs.hasMoreElements()) {
                // 获取下一个元素
                URL url = dirs.nextElement();
                // 得到协议的名称
                String protocol = url.getProtocol();
                // 如果是以文件的形式保存在服务器上
                if ("file".equals(protocol)) {
                    // 获取包的物理路径
                    String filePath = URLDecoder.decode(url.getFile(), "UTF-8");
                    // 以文件的方式扫描整个包下的文件 并添加到集合中
                    addClass(classes, filePath, packageName);
                } else if ("jar".equals(protocol)) {
                    // 如果是jar包文件
                    // 定义一个JarFile
                    JarFile jar;
                    try {
                        // 获取jar
                        jar = ((JarURLConnection) url.openConnection()).getJarFile();
                        // 从此jar包 得到一个枚举类
                        Enumeration<JarEntry> entries = jar.entries();
                        // 同样的进行循环迭代
                        while (entries.hasMoreElements()) {
                            // 获取jar里的一个实体 可以是目录 和一些jar包里的其他文件 如META-INF等文件
                            JarEntry entry = entries.nextElement();
                            String name = entry.getName();
                            // 如果是以/开头的
                            if (name.charAt(0) == '/') {
                                // 获取后面的字符串
                                name = name.substring(1);
                            }
                            // 如果前半部分和定义的包名相同
                            if (name.startsWith(packageDirName)) {
                                int idx = name.lastIndexOf('/');
                                // 如果以"/"结尾 是一个包
                                if (idx != -1) {
                                    // 获取包名 把"/"替换成"."
                                    packageName = name.substring(0, idx).replace('/', '.');
                                }
                                // 如果可以迭代下去 并且是一个包
                                if ((idx != -1) || recursive) {
                                    // 如果是一个.class文件 而且不是目录
                                    if (name.endsWith(".class") && !entry.isDirectory()) {
                                        // 去掉后面的".class" 获取真正的类名
                                        String className = name.substring(packageName.length() + 1, name.length() - 6);
                                        try {
                                            // 添加到classes
                                            classes.add(Class.forName(packageName + '.' + className));
                                        } catch (ClassNotFoundException e) {
                                            e.printStackTrace();
                                        }
                                    }
                                }
                            }
                        }
                    } catch (IOException e) {
                        e.printStackTrace();
                    }
                }
            }
        } catch (IOException e) {
            e.printStackTrace();
        }

        return classes;
    }

    public void addClass(Set<Class<?>> classes, String filePath, String packageName) throws Exception {
        File[] files = new File(filePath).listFiles(file -> (file.isFile() && file.getName().endsWith(".class")) || file.isDirectory());
        assert files != null;
        for (File file : files) {
            String fileName = file.getName();
            if (file.isFile()) {
                String classsName = fileName.substring(0, fileName.lastIndexOf("."));
                if (!packageName.isEmpty()) {
                    classsName = packageName + "." + classsName;
                }
                doAddClass(classes, classsName);
            }

        }
    }

    public void doAddClass(Set<Class<?>> classes, final String classsName) throws Exception {
        ClassLoader classLoader = new ClassLoader() {
            @Override
            public Class<?> loadClass(String name) throws ClassNotFoundException {
                return super.loadClass(name);
            }
        };
        classes.add(classLoader.loadClass(classsName));
    }


    public <A extends Annotation> Set<Class<?>> getAnnotationClasses(String packageName, Class<A> annotationClass) throws Exception {

        //找用了annotationClass注解的类
        Set<Class<?>> controllers = new HashSet<>();
        Set<Class<?>> clsList = getClasses(packageName);
        if (clsList != null && clsList.size() > 0) {
            for (Class<?> cls : clsList) {
                if (cls.getAnnotation(annotationClass) != null) {
                    controllers.add(cls);
                }
            }
        }
        return controllers;
    }

    public static void main(String[] args) throws Exception {
        Set<Class<?>> set = new Scanner().getAnnotationClasses("com.xxx", ControllerAdvice.class);
        System.out.println(set);
    }

}


相关文章

  • @ControllerAdvice 不生效踩坑

    背景 遇到一个bug,同一个接口,在开发环境和其他环境,期望得到的异常请求结果居然不同,但每个环境的代码都是一致的...

  • 记一次SpringBoot 单测踩坑记录

    单元测试踩坑记录: 使用Mockito.when() 时mock总是不生效 (1)问题when(commonSer...

  • Gson @SerializedName 别名

    踩坑。 使用 Gson @SerializedName 取别名后,原来定义的变量名称将不会生效,如要两者同时生效使用:

  • Flutter iOS混编踩坑

    记录踩坑 一、Flutter 和 iOS 混编,修改Dart代码,原生运行不生效 这一点主要是在Flutter3这...

  • 解决MySQL:too many connection

    1.修改配置文件my.cnf Window系统下得以解决 Ubuntu系统配置不生效 2.踩坑:Druid连接池的...

  • D1094:踩坑的价值最大化

    是人就会踩坑,不踩坑理论上就不属于人类,踩坑是人之常情,能回头站在坑边反思,才是对踩过的的坑价值最大化的体现,要不...

  • 交互设计师所要避免的几个坑

    前言 工作中难免会踩到几个坑,即使现在不踩以后还会踩,只有踩过才会深刻记住,踩过说明爱过!但是踩过的坑必须把坑填满...

  • Mac 使用踩坑之旅-.bashrc文件不生效

    Mac下配置环境变量配置不生效.bashrc文件每次打开终端都需要source的问题 先了解Mac环境变量加载顺序...

  • Retrofit Https踩坑记录

    Retrofit Https踩坑记录 前言 新司机上路,坑多,本文重点是踩坑,不详细讲retrofit用法,本文不...

  • vue 踩坑记 scoped + v-html 不生效

    动态生成的内容 即 通过 v-html 创建的 DOM 内容不受 scoped 样式影响。这个时候可以用深度选择器...

网友评论

      本文标题:@ControllerAdvice 不生效踩坑

      本文链接:https://www.haomeiwen.com/subject/abbcjktx.html