当前位置: 首页 > news >正文

SearchClassUtil

路径扫描工具SearchClassUtil,用于扫描指定包(XXXX)下的所有.class文件,并将它们的全限定类名(如tomcat.SearchClassUtil)收集到列表中返回。该工具使用递归文件遍历和反射机制,是实现 Spring 框架组件扫描、Servlet 容器类加载等功能的基础。

完整代码:

package tomcat;import java.io.File;
import java.util.ArrayList;
import java.util.List;/*
* 扫描包下的文件,获取全路径名
* */
public class SearchClassUtil {public static List<String> classPaths = new ArrayList<String>();public static List<String> searchClass(){//需要扫描的包名String basePack = "tomcat";//将获取到的包名转换为路径String classPath = SearchClassUtil.class.getResource("/").getPath();basePack =  basePack.replace(".", File.separator);String searchPath = classPath + basePack;doPath(new File(searchPath),classPath);//这个时候我们已经得到了指定包下所有的类的绝对路径了。我们现在利用这些绝对路径和java的反射机制得到他们的类对象return classPaths;}/*** 该方法会得到所有的类,将类的绝对路径写入到classPaths中* @param file*/private static void doPath(File file,String classpath) {if (file.isDirectory()) {//文件夹//文件夹我们就递归File[] files = file.listFiles();for (File f1 : files) {doPath(f1,classpath);}} else {//标准文件//标准文件我们就判断是否是class文件if (file.getName().endsWith(".class")) {String path = file.getPath().replace(classpath.replace("/","\\").replaceFirst("\\\\",""),"").replace("\\",".").replace(".class","");//如果是class文件我们就放入我们的集合中。classPaths.add(path);}}}public static void main(String[] args) {List<String> classes = SearchClassUtil.searchClass();for (String s: classes) {System.out.println(s);}}
}

代码逐行解释

1. 类定义与成员变量

public class SearchClassUtil {public static List<String> classPaths = new ArrayList<String>();
}
  • 功能:定义工具类,使用静态列表存储扫描到的类名。
  • 注意点:静态变量会在多次调用时累积结果,需手动清空或改进为非静态设计。

2. searchClass() 方法

public static List<String> searchClass(){String basePack = "tomcat";String classPath = SearchClassUtil.class.getResource("/").getPath();basePack =  basePack.replace(".", File.separator);String searchPath = classPath + basePack;doPath(new File(searchPath),classPath);return classPaths;
}
  • 功能:入口方法,初始化扫描参数并启动递归扫描
  • 步骤分解
    1. 设置扫描包名basePack = "tomcat" 硬编码扫描目标包。
    2. 获取类路径根目录
      SearchClassUtil.class.getResource("/").getPath()
      
       
      • 返回当前类所在的类路径根(如/target/classes/或 JAR 包路径)。
      • 示例:若类位于tomcat/SearchClassUtil.class,则返回/path/to/classes/
    3. 转换包名为路径格式
      basePack.replace(".", File.separator)
      
       
      • 将包名(如tomcat.util)转换为文件路径(如tomcat/util)。
      • 使用File.separator确保跨平台兼容性(Windows 为\,Linux 为/)。
    4. 拼接完整搜索路径
      String searchPath = classPath + basePack;
      
       
      • 示例:拼接后为/path/to/classes/tomcat/
    5. 启动递归扫描
      doPath(new File(searchPath), classPath);
      
       
      • 传递搜索目录和类路径根目录。

3. doPath(File file, String classpath) 方法

private static void doPath(File file, String classpath) {if (file.isDirectory()) {File[] files = file.listFiles();for (File f1 : files) {doPath(f1, classpath);}} else {if (file.getName().endsWith(".class")) {String path = file.getPath().replace(classpath.replace("/","\\").replaceFirst("\\\\",""),"").replace("\\",".").replace(".class","");classPaths.add(path);}}
}
  • 功能:递归遍历文件系统,提取类名并存储到列表。
  • 逻辑分解
    1. 目录处理
      if (file.isDirectory()) { ... }
      
       
      • 递归调用doPath处理子文件 / 目录。
    2. 文件处理
      if (file.getName().endsWith(".class")) { ... }
      
       
      • 筛选.class文件。
    3. 路径处理
      String path = file.getPath().replace(classpath.replace("/","\\").replaceFirst("\\\\",""),"").replace("\\",".").replace(".class","");
      
       
      • 复杂替换逻辑详解
        1. classpath.replace("/","\\").replaceFirst("\\\\","")
          • 将类路径(如/path/to/classes/)转换为系统路径格式(如path\to\classes\)。
          • 示例:输入/C:/path/to/classes/ → 输出C:\path\to\classes\
        2. file.getPath().replace(...)
          • 移除类路径前缀,保留相对路径。
          • 示例:原路径C:\path\to\classes\tomcat\MyClass.class → tomcat\MyClass.class
        3. replace("\\",".")
          • 将路径分隔符转换为.,形成类名格式。
          • 示例tomcat\MyClass.class → tomcat.MyClass.class
        4. replace(".class","")
          • 移除文件扩展名,得到全限定类名。
          • 示例tomcat.MyClass.class → tomcat.MyClass

4. main(String[] args) 方法

public static void main(String[] args) {List<String> classes = SearchClassUtil.searchClass();for (String s: classes) {System.out.println(s);}
}
  • 功能:测试工具类,打印扫描到的类名。
  • 执行流程
    1. 调用searchClass()方法获取类名列表。
    2. 遍历列表并打印每个类名。

关键技术细节

1. 类路径资源获取

SearchClassUtil.class.getResource("/").getPath()
  • 工作原理
    • getResource("/")返回类路径根目录的URL对象。
    • .getPath()URL转换为文件系统路径。
  • 注意点
    • 路径格式可能包含协议前缀(如file:/),但后续替换逻辑会处理此问题。
    • 在 JAR 包环境中,可能返回jar:file:/path/to/app.jar!/格式,此时该代码会失效。

2. 路径处理逻辑

file.getPath().replace(classpath.replace("/","\\").replaceFirst("\\\\",""),"")
  • 设计意图
    • 移除类路径前缀,保留相对路径。
  • 存在问题
    • 跨平台兼容性:混用/\\作为路径分隔符,可能在非 Windows 系统上出错。
    • 特殊路径处理:若类路径包含特殊字符(如空格),替换逻辑可能失效。

3. 静态列表的线程安全

public static List<String> classPaths = new ArrayList<String>();

4.classpath.replace("/","\\").replaceFirst("\\\\","") 详解

 这段代码的核心目标是将类路径字符串转换为系统路径格式,并移除可能存在的前导路径分隔符,以便后续正确拼接和替换路径。

 假设初始 classpath 值为:

String classpath = "/C:/workspace/project/target/classes/";
  1. 第一步:classpath.replace("/", "\\")

    • 将所有斜杠 / 替换为反斜杠 \
    • 结果:"\C:\workspace\project\target\classes\"
  2. 第二步:.replaceFirst("\\\\", "")

    • 使用正则表达式 \\\\(对应 Java 字符串中的 "\\",即正则中的 \)匹配第一个反斜杠
    • 将其替换为空字符串
    • 结果:"C:\workspace\project\target\classes\"

5. dopath递归

假设目录结构如下:

classes/
└── tomcat/├── util/│   ├── StringUtil.class│   └── FileUtil.class└── SearchClassUtil.class

递归执行步骤:

  1. 初始调用doPath("classes/tomcat/", "classes/")

    • 处理目录 tomcat/
    • 递归调用子项:util/ 和 SearchClassUtil.class
  2. 处理 util/ 目录

    • 递归调用子项:StringUtil.class 和 FileUtil.class
  3. 处理 StringUtil.class

    • 路径转换:classes/tomcat/util/StringUtil.class → tomcat.util.StringUtil
    • 加入结果列表
  4. 处理 FileUtil.class

    • 路径转换:classes/tomcat/util/FileUtil.class → tomcat.util.FileUtil
    • 加入结果列表
  5. 返回处理 SearchClassUtil.class

    • 路径转换:classes/tomcat/SearchClassUtil.class → tomcat.SearchClassUtil
    • 加入结果列表

相关文章:

  • 在 Ubuntu 20.04 中使用 init.d 或者systemd实现开机自动执行脚本
  • YOLOv3深度解析:多尺度特征融合与实时检测的里程碑
  • 淘宝扭蛋机系统开发前景分析:解锁电商娱乐化新蓝海
  • 执行apt-get update 报错ModuleNotFoundError: No module named ‘apt_pkg‘的解决方案汇总
  • 文件系统交互实现
  • 特斯拉虚拟电厂:能源互联网时代的分布式革命
  • NX二次开发C#---遍历当前工作部件实体并设置颜色
  • 来一个复古的技术FTP
  • 交叉熵损失函数,KL散度, Focal loss
  • PHP:经典编程语言在新时代的持续活力与演进
  • 中exec()函数因$imagePath参数导致的命令注入漏洞
  • 自定义CString类与MFC CString类接口对比
  • 奥运数据可视化:探索数据讲述奥运故事
  • w~深度学习~合集3
  • PyTorch 的 F.scaled_dot_product_attention 返回Nan
  • 三格电子上新了——Modbus转IEC104网关
  • C42-作业练习
  • 速通RocketMQ配置
  • MySQL——3、数据类型
  • YOLOv8在单目向下多车辆目标检测中的应用
  • 贝壳一季度收入增长42%:二手房市场活跃度维持在高位
  • 北京警方:海淀发生小客车刮碰行人事故4人受伤,肇事司机已被查获
  • 商务部回应稀土出口管制问题
  • 腾讯一季度净利增14%:AI直接拉动广告收入增长,王者荣耀流水创新高
  • 有人倒卖试运营门票?上海乐高乐园:这些票存在无法入园风险
  • 现场丨在胡适施蛰存等手札与文献间,再读百年光华