最近做了一个技改项目,解除两个应用(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