美文网首页
查某一个包下所有同名的类(包名不完全一样)

查某一个包下所有同名的类(包名不完全一样)

作者: 安然在路上 | 来源:发表于2022-02-10 16:10 被阅读0次

最近做了一个技改项目,解除两个应用(A,B)之间的相互依赖,解决方法是把A需要引用的B的类复制一份到A中(全限定类名和B中的类完全一样),B同理.
上线之后,发现有的类移动错了,本来A中应该是com.hupan.passion.Adto被我移成了com.hupan.Adto,同时com.hupan.passion.Adto也存在.因为idea没有直接排查所有同名类的按钮,所以用代码跑了一遍,代码如下(百度了一下,删掉了一下),亲测,可用

package com.fpx.fb4.dispatch;


import java.io.File;
import java.io.FileFilter;
import java.io.IOException;
import java.net.JarURLConnection;
import java.net.URL;
import java.net.URLDecoder;
import java.util.*;
import java.util.jar.JarEntry;
import java.util.jar.JarFile;


/**
 * @Author:hupan
 * @Date:2022/2/10 10:46
 */
public class ClassUtil {
    public static void main(String[] args) throws Exception{
        List<Class> classes = ClassUtil.getAllClassByInterface("com.fpx.fb4.dispatch");
        Map<String,Integer> map =new HashMap<>();
        for (Class clas :classes) {
            String className = clas.getName();
            String substring = className.substring(className.lastIndexOf(".") + 1);
            map.put(substring,map.get(substring)==null?1:map.get(substring)+1);
        }
        for(String key:map.keySet()){
            if(map.get(key)>1){
                System.out.println("Value:"+map.get(key) +" "+ "key:"+key);
            }
        }
    }
    /**
     * 取得某个接口下所有实现这个接口的类
     * */
    public static List<Class> getAllClassByInterface(String packageName) {
        List<Class>  returnClassList = null;
            // 获取当前包下以及子包下所以的类
            List<Class<?>> allClass = getClasses(packageName);
            if(allClass != null) {
                returnClassList = new ArrayList<Class>();
                for(Class classes : allClass) {
                    returnClassList.add(classes);
                }
            }
        return returnClassList;
    }

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

        //第一个class类的集合
        List<Class<?>> classes = new ArrayList<Class<?>>();
        //是否循环迭代
        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");
                    //以文件的方式扫描整个包下的文件 并添加到集合中
                    findAndAddClassesInPackageByFile(packageName, filePath, recursive, classes);
                }
            }
        } catch (IOException e) {
            e.printStackTrace();
        }
        return classes;
    }

    /**
     * 以文件的形式来获取包下的所有Class
     * @param packageName
     * @param packagePath
     * @param recursive
     * @param classes
     */
    public static void findAndAddClassesInPackageByFile(String packageName, String packagePath, final boolean recursive, List<Class<?>> classes){
        //获取此包的目录 建立一个File
        File dir = new File(packagePath);
        //如果不存在或者 也不是目录就直接返回
        if (!dir.exists() || !dir.isDirectory()) {
            return;
        }
        //如果存在 就获取包下的所有文件 包括目录
        File[] dirfiles = dir.listFiles(new FileFilter() {
            //自定义过滤规则 如果可以循环(包含子目录) 或则是以.class结尾的文件(编译好的java类文件)
            public boolean accept(File file) {
                return (recursive && file.isDirectory()) || (file.getName().endsWith(".class"));
            }
        });
        //循环所有文件
        for (File file : dirfiles) {
            //如果是目录 则继续扫描
            if (file.isDirectory()) {
                findAndAddClassesInPackageByFile(packageName + "." + file.getName(),
                        file.getAbsolutePath(), recursive, classes);
            } else {
                //如果是java类文件 去掉后面的.class 只留下类名
                String className = file.getName().substring(0, file.getName().length() - 6);
                try {
                    //添加到集合中去
                    classes.add(Class.forName(packageName + '.' + className));
                } catch (ClassNotFoundException e) {
                    e.printStackTrace();
                }
            }
        }
    }
}

输出:

Value:2 key:OtaskSkuDTO

相关文章

  • 查某一个包下所有同名的类(包名不完全一样)

    最近做了一个技改项目,解除两个应用(A,B)之间的相互依赖,解决方法是把A需要引用的B的类复制一份到A中(全限定类...

  • 第三章③导入【同名不同包】的类

    同一个包下,当代码前已经导入了一个包下的类 当需要导入【同名不同包】的类: 要显示的通过 包名.包.类java.s...

  • 包 Package和import

    包的引入是为了区别同名类的。不同的包下面可以有相同名称的类。包类似于身份证号码 注意: 1、package语句...

  • Android知识体系之代码混淆

    混淆规则: 注意一颗星表示只是保持该包下的类名,而子包下的类名还是会被混淆;两颗星表示把本包及子包下的类名都会被保...

  • 3.use-fault-filters

    配置自动扫描的包在当前包下面所有的类,和当前包下所有子包中所有的类@Component@Service@Contr...

  • 同名类的加载顺序

    maven在遇到同名类时,加载顺序如何?我们做实验测试下。 出现同名类的情况如下,不同包下的MyApi类完全一样:...

  • Java中常用的包,类,接口

    用package声明包,用import导包* 表示一个通配符---导入当前包下的所有的类,但是不包括子包下的类--...

  • scala(八) 面向对象

    Scala包 基本语法:package 包名 Scala包的三大作用(和Java一样)1.区分相同名字的类2.当类...

  • Java中的面向对象的简单介绍(五)包和修饰符以及内部类

    包 特点: 可以有多层 不同包下面的文件名可以重复 包的声明必须要在第一行 不同包下的相互访问 相同包下的类可以直...

  • 13. 包(package)

    kotlin 中的包其实就是命名空间。不同的包下,可以有相同的类。本包中调用本类可以直接写类名,调用其他包的类需要...

网友评论

      本文标题:查某一个包下所有同名的类(包名不完全一样)

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