前几天项目中让扫描出所有使用Restful API的方法。刚开始还想着用python过滤关键字来查找的,后来想想可以使用反射来搞的。主要包含以下三个步骤:

  1. 根据包名解析包的具体路径
  2. 查找指定包下指定注解的类
  3. 在上一步骤中得到的类中,依次扫描包含指定注解的方法

想着写着工具类的形式,代码结构如下:

public class AnnotationScannerUtils {private static final Logger logger = LoggerFactory.getLogger(AnnotationScannerUtils.class);private static final String EXT = "class/*** 根据包名获取包的URL* @param pkgName com.demo.controller* @return*/public static String getPkgPath(String pkgName){String pkgDirName = pkgName.replace('.', File.separatorChar);URL url = Thread.currentThread().getContextClassLoader().getResource(pkgDirName);return url == null ? null : url.getFile();}/*** 获取指定包下所有类对象的集合* @param pkgName 包名(com.demo.controller)* @param pkgPath 包路径(/Users/xxx/workspace/java/project/out/production/classes/com/demo/controller)* @param recursive 是否递归遍历子目录* @return 类集合*/public static Set<Class<?>> scanClasses(String pkgName, String pkgPath, final boolean recursive){Set<Class<?>> classesSet = new HashSet<>();Collection<File> allClassFile = getAllClassFile(pkgPath, recursive);for (File curFile : allClassFile){try {classesSet.add(getClassObj(curFile, pkgPath, pkgName));} catch (ClassNotFoundException e) {logger.error("load class fail", e);}}return classesSet;}/*** 获取指定包下包含指定注解的所有类对象的集合* @param pkgName 包名(com.demo.controller)* @param pkgPath 包路径(/Users/xxx/workspace/java/project/out/production/classes/com/demo/controller)* @param recursive 是否递归遍历子目录* @param targetAnnotations 指定注解* @return 以注解和对应类集合构成的键值对*/public static Map<Class<? extends Annotation>, Set<Class<?>>> scanClassesByAnnotations(String pkgName, String pkgPath, final boolean recursive, List<Class<? extends Annotation>> targetAnnotations){Map<Class<? extends Annotation>, Set<Class<?>>> resultMap = new HashMap<>(16);Collection<File> allClassFile = getAllClassFile(pkgPath, recursive);for (File curFile : allClassFile){try {Class<?> curClass = getClassObj(curFile, pkgPath, pkgName);for (Class<? extends Annotation> annotation : targetAnnotations){if (curClass.isAnnotationPresent(annotation)){if (!resultMap.containsKey(annotation)){resultMap.put(annotation, new HashSet<Class<?>>());}resultMap.get(annotation).add(curClass);}}} catch (ClassNotFoundException e) {logger.error("load class fail", e);}}return resultMap;}/*** 加载类* @param file* @param pkgPath* @param pkgName* @return* @throws ClassNotFoundException*/private static Class<?> getClassObj(File file, String pkgPath, String pkgName) throws ClassNotFoundException{// 考虑class文件在子目录中的情况String absPath = file.getAbsolutePath().substring(0, file.getAbsolutePath().length() - EXT.length() - 1);String className = absPath.substring(pkgPath.length()).replace(File.separatorChar, '.');className = className.startsWith(".") ? pkgName + className : pkgName + "." + className;return Thread.currentThread().getContextClassLoader().loadClass(className);}/*** 遍历指定目录下所有扩展名为class的文件* @param pkgPath 包目录* @param recursive 是否递归遍历子目录* @return*/private static Collection<File> getAllClassFile(String pkgPath, boolean recursive){File fPkgDir = new File(pkgPath);if (!(fPkgDir.exists() && fPkgDir.isDirectory())){logger.error("the directory to package is empty: {}", pkgPath);return null;}return FileUtils.listFiles(fPkgDir, new String[]{EXT}, recursive);}/*** 查找指定注解的Method* @param classes 查找范围* @param targetAnnotations 指定的注解* @return 以注解和对应Method类集合构成的键值对*/public static Map<Class<? extends Annotation>, Set<Method>> scanMethodsByAnnotations(Set<Class<?>> classes,List<Class<? extends Annotation>> targetAnnotations){Map<Class<? extends Annotation>, Set<Method>> resultMap = new HashMap<>(16);for (Class<?> cls : classes){Method[] methods = cls.getMethods();for (Class<? extends Annotation> annotation : targetAnnotations){for (Method method : methods){if (method.isAnnotationPresent(annotation)){if (!resultMap.containsKey(annotation)){resultMap.put(annotation, new HashSet<Method>());}resultMap.get(annotation).add(method);}}}}return resultMap;}
}
复制代码

具体使用时,可根据具体情况在原方法上二次开发。如果是直接调用,可以实现扫描包含指定注解的类和方法:

    public static void main(String[] args) {String pkgName = "com.demo.controller";String pkgPath = getPkgPath(pkgName);logger.info("pkgPath is {}", pkgName);// 查找包含RestController和Controller注解的类Map<Class<? extends Annotation>, Set<Class<?>>> classesMap = scanClassesByAnnotations(pkgName, pkgPath, true,Arrays.asList(RestController.class, Controller.class));if (classesMap.size() == 0){logger.error("Not exists any class in {} with the specified annotation", pkgPath);return;}Set<Class<?>> classSet = new HashSet<>();classesMap.forEach((k, v) -> {logger.info("get {} classes with {}", v.size(), k.getSimpleName());classSet.addAll(v);});// 查找包含GetMapping和PostMapping注解的MethodMap<Class<? extends Annotation>, Set<Method>> methodMap = scanMethodsByAnnotations(classSet, Arrays.asList(GetMapping.class, PostMapping.class));if (methodMap.size() == 0){logger.error("Not exists any method with the specified annotation");return;}methodMap.forEach((k, v) -> {StringBuilder sb = new StringBuilder();v.forEach(method -> sb.append(method.getName()+", "));logger.info(k.getSimpleName() + ": " + sb.toString());});}--------------------------output-------------------------
01-20 15:06:02.293 [ INFO] [            m.i.u.AnnotationScannerUtils: 29] - pkgPath is com.demo.controller
01-20 15:06:02.363 [ INFO] [            m.i.u.AnnotationScannerUtils: 41] - get 5 classes with RestController
01-20 15:06:02.374 [ INFO] [            m.i.u.AnnotationScannerUtils: 41] - get 2 classes with Controller
01-20 15:06:02.388 [ INFO] [            m.i.u.AnnotationScannerUtils: 55] - PostMapping: login, addFavorite, addUser, logout,
01-20 15:06:02.388 [ INFO] [            m.i.u.AnnotationScannerUtils: 55] - GetMapping: webSiteInfo, info, getCategories, favoritesList, getReportByCityId, queryUserById, getReportByCityName, queryBookById,
复制代码

java随笔-扫描使用指定注解的类与方法相关推荐

  1. java扫描指定package注解_java随笔-扫描使用指定注解的类与方法

    前几天项目中让扫描出所有使用Restful API的方法.刚开始还想着用python过滤关键字来查找的,后来想想可以使用反射来搞的.主要包含以下三个步骤: 根据包名解析包的具体路径 查找指定包下指定注 ...

  2. java 中成员变量的名字和类的方法的名字可以一样.

    java 中成员变量的名字和类的方法的名字可以一样. public class fourone {public static void main(String[] args) {int first[] ...

  3. java扫描指定package注解_java获取包下被指定注解的类

    方案一: 采用reflections 框架(此框架依赖com.google.guava) 2.项目依赖 org.reflections reflections 0.9.11 com.google.gu ...

  4. java 查找注解_Java利用反射如何查找使用指定注解的类详解

    前言 最近有些空,想自己写个跟spring里的注解一样的注解来用,然后希望能找到使用了自己写了注解的类,下面来介绍一下实现方法 声明,下面代码是没看过spring源码写的,基本上都是网上找的博客,整理 ...

  5. java 获取所有带指定注解的类名_SXT DAY023 反射和注解

    1. 反射机制介绍_Class对象获取 反射机制是 Java 的动态性之一 动态语言:在程序运行时,可以改变程序的结构或变量的 类型. 反射机制的常见作用 动态的加载类.动态的获取类的信息(属性,方法 ...

  6. Java调用浏览器打开指定页面的5种方法(最全)

    package com.xbz;import java.awt.*; import java.lang.reflect.Method; import java.net.URI; import java ...

  7. java中file类乱,【JAVA SE基础篇】47.file类的方法

    1.file类的方法 1.getName():获取文件名 2.getPath():获取文件路径,设定是相对路径得到的就是相对路径,绝对路径同里 3.getAbsolutePath():获取文件绝对路径 ...

  8. Java中的isnan函数_Java Float类isNaN()方法与示例

    语法:public boolean isNaN (); public static boolean isNaN(float value); float 类isNaN()方法isNaN()方法在java ...

  9. java - 随机生成人名/指定位数字符串/手机号 --工具方法

    我是一个B话不多的人 1.随机生成人名/指定位数字符串 package ls.example.test.util;import java.io.*; import java.util.Random; ...

最新文章

  1. RANSAC算法做直线拟合
  2. s:property的用法
  3. php如何获取服务器时间_php如何获取当前日期
  4. Mendix:低代码平台四大常见用例开发
  5. 分页缓冲池占用很高怎么解决_一次线上服务高 CPU 占用优化实践
  6. hnu 暑期实训之7还是7 to_string的实现方法
  7. 非金钱激励员工的108种手段【转-摘】
  8. 重磅!泰晤士发布重量级学科排名,90所中国大陆高校上榜!
  9. hane WIN nfs配置
  10. 计算机组成原理bzc指令,计算机组成原理课程设计-09级.ppt
  11. 论文阅读:Reasoning about Object Affordances in a Knowledge Base Representation
  12. 【回炉重造系列】之Spring AOP简介
  13. Java零基础学习记录09(飞机躲避炮弹游戏实现)
  14. 基于FPGA 的TF卡 UHS-I 方式存储
  15. html指定区域的背景颜色,JS实现点击颜色块切换指定区域背景颜色的方法
  16. 关于SetCapture和GetCapture
  17. 360°全景影像建库流程
  18. 计算机控制直流电机闭环调速实验报告,PID控制电机实验报告
  19. Error instantiating interface com.dao.UserDao with invalid types () or values (). Cause: java.lang.N
  20. 一般可行性研究报告的主要内容和要求

热门文章

  1. ofstream清空文件内容_回收站被删除的文件怎么恢复 回收站清空了怎么恢复
  2. python 开发工具_Python开发工具之神兵利器
  3. C++中栈区 堆区 常量区
  4. unittest无法输出测试结果至文件
  5. 亲密关系-【沟通目标】如何建立有效的谈话目标
  6. 一招教你在linux服务器配置Jenkins持续集成神器
  7. 狼来了!第一批90后测试员已经开始经历测试行业求职危机……
  8. 机器学习实战—朴素贝叶斯及要点注解
  9. 如何从文件名字符串中获取文件扩展名_Linux操作系统:文件系统的功能和命名...
  10. scala解析csv文件写入mysql_scala实战之spark源码修改(能够将DataFrame按字段增量写入mysql数据表)...