java获取包下被指定注解的类过程解析

作者:ggband 时间:2023-08-08 11:12:27 

方案一: 采用reflections 框架(此框架依赖com.google.guava)

1、reflections框架地址:https://github.com/ronmamo/reflections

2、项目依赖


<dependency>
     <groupId>org.reflections</groupId>
     <artifactId>reflections</artifactId>
     <version>0.9.11</version>
   </dependency>

<dependency> <groupId>com.google.guava</groupId> <artifactId>guava</artifactId>
     <version>21.0</version>
   </dependency>

3、实现代码


//入参 要扫描的包名
Reflections f = new Reflections("com.ggband.netty.execute.command");
//入参 目标注解类
Set<Class<?>> set = f.getTypesAnnotatedWith(Cmd.class);

方案二: 采用ClassLoader扫描

1、实现代码


package com.ggband.netty;

import java.io.File;
import java.io.IOException;
import java.lang.annotation.Annotation;
import java.net.JarURLConnection;
import java.net.URL;
import java.net.URLDecoder;
import java.util.Enumeration;
import java.util.HashSet;
import java.util.Set;
import java.util.jar.JarEntry;
import java.util.jar.JarFile;

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;
 }

}

2、使用:


Set<Class<?>> set = new Scanner().getAnnotationClasses("com.ggband.netty.execute.command", Cmd.class);

扩充:现在就可以实现自己的业务了,比如 扫描com.ggband.netty.execute.command包下被Cmd注解的类 得到Cmd注解value和被注解类的实例


Map<String, Command> beanContainer = new HashMap<>();
   try {
     //@1 采用reflections 框架(此框架依赖com.google.guava)
//       Reflections f = new Reflections("com.ggband.netty.execute.command");
//       Set<Class<?>> set = f.getTypesAnnotatedWith(Cmd.class);
     //@2 采用ClassLoader扫描
     Set<Class<?>> set = new Scanner().getAnnotationClasses("com.ggband.netty.execute.command", Cmd.class);
     for (Class<?> c : set) {
       Object bean = c.newInstance();
       Cmd annotation = c.getAnnotation(Cmd.class);
       beanContainer.put(Arrays.toString(annotation.value()), (Command) bean);
     }
   } catch (Exception e) {
     e.printStackTrace();
   }

来源:https://www.cnblogs.com/ggband/p/11668879.html

标签:java,获取,注解,类
0
投稿

猜你喜欢

  • Unity实现透视滑动列表

    2022-04-10 20:15:40
  • 详解Spring 中如何控制2个bean中的初始化顺序

    2023-05-06 13:43:18
  • Java多线程的具体介绍与使用笔记小结

    2023-01-17 05:23:13
  • android实现扑克卡片翻转

    2021-10-19 08:08:35
  • Java 实战项目锤炼之网上商城系统的实现流程

    2022-06-26 17:21:47
  • Java将字符串转化为数组的两种方法

    2021-07-07 20:33:49
  • Zookeeper和Eureka哪个更好?

    2023-11-10 02:57:35
  • Guava - 并行编程Futures详解

    2022-04-28 23:16:34
  • C# 9.0新特性——扩展方法GetEnumerator支持foreach循环

    2021-08-27 22:38:09
  • C语言折半查找法的超详细讲解

    2022-10-26 19:33:16
  • C#使用Effects给图片增加阴影效果

    2023-09-09 14:50:18
  • Java编写实现坦克大战小游戏

    2023-11-24 09:15:34
  • Mybatis环境搭建及文件配置过程解析

    2021-07-04 22:37:03
  • Java字符串split使用方法代码实例

    2023-02-06 18:55:31
  • 深入分析Android系统中SparseArray的源码

    2022-09-16 13:54:42
  • Golang+Android基于HttpURLConnection实现的文件上传功能示例

    2023-10-27 06:19:46
  • C#算法之散列表

    2022-07-30 19:05:52
  • Java开发人员需知的十大戒律

    2023-09-17 07:33:50
  • Spring事务事件监控的实现

    2022-11-28 06:19:13
  • java中使用interrupt通知线程停止详析

    2023-09-03 11:41:26
  • asp之家 软件编程 m.aspxhome.com