com.base4j.util.ClassUtil Maven / Gradle / Ivy
The newest version!
package com.base4j.util;
import java.io.File;
import java.io.FileFilter;
import java.io.IOException;
import java.lang.annotation.Annotation;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.lang.reflect.Modifier;
import java.lang.reflect.Proxy;
import java.net.URI;
import java.net.URL;
import java.util.ArrayList;
import java.util.Collections;
import java.util.Date;
import java.util.Enumeration;
import java.util.HashSet;
import java.util.List;
import java.util.Locale;
import java.util.Set;
import java.util.jar.JarEntry;
import java.util.jar.JarFile;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import com.base4j.util.exceptions.UtilException;
import com.base4j.util.lang.BasicType;
import com.base4j.util.lang.Filter;
import com.base4j.util.lang.Singleton;
/**
* 类工具类 1、扫描指定包下的所有类
* 参考 http://www.oschina.net/code/snippet_234657_22722
*
* @author xxx
*/
public class ClassUtil {
private final static Logger log = LoggerFactory.getLogger(ClassUtil.class);
private ClassUtil() {
// 静态类不可实例化
}
/**
* 查找方法
*
* @param clazz 类
* @param methodName 方法名
* @param paramTypes 参数类型
* @return 方法
*/
public static Method findMethod(Class> clazz, String methodName, Class>... paramTypes) {
try {
return clazz.getMethod(methodName, paramTypes);
} catch (NoSuchMethodException ex) {
return findDeclaredMethod(clazz, methodName, paramTypes);
}
}
/**
* 查找所有方法
*
* @param clazz 类
* @param methodName 方法名
* @param paramTypes 参数类型
* @return Method
*/
public static Method findDeclaredMethod(Class> clazz, String methodName, Class>... paramTypes) {
try {
return clazz.getDeclaredMethod(methodName, paramTypes);
} catch (NoSuchMethodException ex) {
if (clazz.getSuperclass() != null) {
return findDeclaredMethod(clazz.getSuperclass(), methodName, paramTypes);
}
return null;
}
}
/**
* 获得对象数组的类数组
*
* @param objects 对象数组
* @return 类数组
*/
public static Class>[] getClasses(Object... objects) {
Class>[] classes = new Class>[objects.length];
for (int i = 0; i < objects.length; i++) {
classes[i] = objects[i].getClass();
}
return classes;
}
/**
* 扫面该包路径下所有class文件
*
* @return 类集合
*/
public static Set> scanPackage() {
return scanPackage(StrUtil.EMPTY, null);
}
/**
* 扫面该包路径下所有class文件
*
* @param packageName 包路径 com | com. | com.abs | com.abs.
* @return 类集合
*/
public static Set> scanPackage(String packageName) {
return scanPackage(packageName, null);
}
/**
* 扫描指定包路径下所有包含指定注解的类
*
* @param packageName 包路径
* @param annotationClass 注解类
* @return 类集合
*/
public static Set> scanPackageByAnnotation(String packageName, final Class extends Annotation> annotationClass) {
return scanPackage(packageName, new ClassFilter() {
@Override
public boolean accept(Class> clazz) {
return clazz.isAnnotationPresent(annotationClass);
}
});
}
/**
* 扫描指定包路径下所有指定类的子类
*
* @param packageName 包路径
* @param superClass 父类
* @return 类集合
*/
public static Set> scanPackageBySuper(String packageName, final Class> superClass) {
return scanPackage(packageName, new ClassFilter() {
@Override
public boolean accept(Class> clazz) {
return superClass.isAssignableFrom(clazz) && !superClass.equals(clazz);
}
});
}
/**
* 扫面包路径下满足class过滤器条件的所有class文件,
* 如果包路径为 com.abs + A.class 但是输入 abs会产生classNotFoundException
* 因为className 应该为 com.abs.A 现在却成为abs.A,此工具类对该异常进行忽略处理,有可能是一个不完善的地方,以后需要进行修改
*
* @param packageName 包路径 com | com. | com.abs | com.abs.
* @param classFilter class过滤器,过滤掉不需要的class
* @return 类集合
*/
public static Set> scanPackage(String packageName, ClassFilter classFilter) {
if (StrUtil.isBlank(packageName)) {
packageName = StrUtil.EMPTY;
}
log.debug("Scan classes from package [{}]...", packageName);
packageName = getWellFormedPackageName(packageName);
final Set> classes = new HashSet>();
for (String classPath : getClassPaths(packageName)) {
// bug修复,由于路径中空格和中文导致的Jar找不到
classPath = UrlUtil.decode(classPath, CharsetUtil.systemCharset());
log.debug("Scan classpath: [{}]", classPath);
// 填充 classes
fillClasses(classPath, packageName, classFilter, classes);
}
// 如果在项目的ClassPath中未找到,去系统定义的ClassPath里找
if (classes.isEmpty()) {
for (String classPath : getJavaClassPaths()) {
// bug修复,由于路径中空格和中文导致的Jar找不到
classPath = UrlUtil.decode(classPath, CharsetUtil.systemCharset());
log.debug("Scan java classpath: [{}]", classPath);
// 填充 classes
fillClasses(classPath, new File(classPath), packageName, classFilter, classes);
}
}
return classes;
}
/**
* 获得指定类中的Public方法名
* 去重重载的方法
*
* @param clazz 类
*/
public static Set getMethodNames(Class> clazz) {
HashSet methodSet = new HashSet();
Method[] methodArray = clazz.getMethods();
for (Method method : methodArray) {
String methodName = method.getName();
methodSet.add(methodName);
}
return methodSet;
}
/**
* 获得指定类过滤后的方法列表
*
* @param clazz 查找方法的类
* @return 过滤后的方法列表
*/
public static List getMethods(Class> clazz) {
return getMethods(clazz, (Filter) null);
}
/**
* 获得指定类过滤后的方法列表
*
* @param clazz 查找方法的类
* @param filter 过滤器
* @return 过滤后的方法列表
*/
public static List getMethods(Class> clazz, Filter filter) {
if (null == clazz) {
return null;
}
Method[] methods = clazz.getMethods();
List methodList;
if (null != filter) {
methodList = new ArrayList<>();
Method filteredMethod;
for (Method method : methods) {
filteredMethod = filter.modify(method);
if (null != filteredMethod) {
methodList.add(method);
}
}
} else {
methodList = CollectionUtil.newArrayList(methods);
}
return methodList;
}
/**
* 获得指定类过滤后的方法列表
*
* @param clazz 查找方法的类
* @param excludeMethods 不包括的方法
* @return 过滤后的方法列表
*/
public static List getMethods(Class> clazz, Method... excludeMethods) {
final HashSet excludeMethodSet = CollectionUtil.newHashSet(excludeMethods);
Filter filter = new Filter() {
@Override
public Method modify(Method method) {
return excludeMethodSet.contains(method) ? null : method;
}
};
return getMethods(clazz, filter);
}
/**
* 获得指定类过滤后的方法列表
*
* @param clazz 查找方法的类
* @param excludeMethodNames 不包括的方法名列表
* @return 过滤后的方法列表
*/
public static List getMethods(Class> clazz, String... excludeMethodNames) {
final HashSet excludeMethodNameSet = CollectionUtil.newHashSet(excludeMethodNames);
Filter filter = new Filter() {
@Override
public Method modify(Method method) {
return excludeMethodNameSet.contains(method.getName()) ? null : method;
}
};
return getMethods(clazz, filter);
}
/**
* 获得ClassPath
*
* @return ClassPath集合
*/
public static Set getClassPathResources() {
return getClassPaths(StrUtil.EMPTY);
}
/**
* 获得ClassPath
*
* @param packageName 包名称
* @return ClassPath路径字符串集合
*/
public static Set getClassPaths(String packageName) {
String packagePath = packageName.replace(StrUtil.DOT, StrUtil.SLASH);
Enumeration resources;
try {
resources = getClassLoader().getResources(packagePath);
} catch (IOException e) {
throw new UtilException(StrUtil.format("Loading classPath [{}] error!", packagePath), e);
}
Set paths = new HashSet();
while (resources.hasMoreElements()) {
paths.add(resources.nextElement().getPath());
}
return paths;
}
/**
* 获得ClassPath
*
* @return ClassPath
*/
public static String getClassPath() {
return getClassPathURL().getPath();
}
/**
* 获得ClassPath URL
*
* @return ClassPath URL
*/
public static URL getClassPathURL() {
return getURL(StrUtil.EMPTY);
}
/**
* 获得资源的URL
*
* @param resource 资源(相对Classpath的路径)
* @return 资源URL
*/
public static URL getURL(String resource) {
return ClassUtil.getClassLoader().getResource(resource);
}
/**
* @return 获得Java ClassPath路径,不包括 jre
*/
public static String[] getJavaClassPaths() {
String[] classPaths = System.getProperty("java.class.path").split(System.getProperty("path.separator"));
return classPaths;
}
/**
* 转换为原始类型
*
* @param clazz 被转换为原始类型的类,必须为包装类型的类
* @return 基本类型类
*/
public static Class> castToPrimitive(Class> clazz) {
if (null == clazz || clazz.isPrimitive()) {
return clazz;
}
BasicType basicType;
try {
basicType = BasicType.valueOf(clazz.getSimpleName().toUpperCase());
} catch (Exception e) {
return clazz;
}
// 基本类型
switch (basicType) {
case BYTE:
return byte.class;
case SHORT:
return short.class;
case INTEGER:
return int.class;
case LONG:
return long.class;
case DOUBLE:
return double.class;
case FLOAT:
return float.class;
case BOOLEAN:
return boolean.class;
case CHAR:
return char.class;
default:
return clazz;
}
}
/**
* @return 当前线程的class loader
*/
public static ClassLoader getContextClassLoader() {
return Thread.currentThread().getContextClassLoader();
}
/**
* 获得class loader
* 若当前线程class loader不存在,取当前类的class loader
*
* @return 类加载器
*/
public static ClassLoader getClassLoader() {
ClassLoader classLoader = getContextClassLoader();
if (classLoader == null) {
classLoader = ClassUtil.class.getClassLoader();
}
return classLoader;
}
/**
* 实例化对象
*
* @param clazz 类名
* @return 对象
*/
@SuppressWarnings("unchecked")
public static T newInstance(String clazz) {
try {
return (T) Class.forName(clazz).newInstance();
} catch (Exception e) {
throw new UtilException(StrUtil.format("Instance class [{}] error!", clazz), e);
}
}
/**
* 实例化对象
*
* @param clazz 类
* @return 对象
*/
public static T newInstance(Class clazz) {
try {
return (T) clazz.newInstance();
} catch (Exception e) {
throw new UtilException(StrUtil.format("Instance class [{}] error!", clazz), e);
}
}
/**
* 实例化对象
*
* @param clazz 类
* @return 对象
*/
public static T newInstance(Class clazz, Object... params) {
if (CollectionUtil.isEmpty(params)) {
return newInstance(clazz);
}
try {
return clazz.getDeclaredConstructor(getClasses(params)).newInstance(params);
} catch (Exception e) {
throw new UtilException(StrUtil.format("Instance class [{}] error!", clazz), e);
}
}
/**
* 加载类
*
* @param
* @param className 类名
* @param isInitialized 是否初始化
* @return 类
*/
@SuppressWarnings("unchecked")
public static Class loadClass(String className, boolean isInitialized) {
Class clazz;
try {
clazz = (Class) Class.forName(className, isInitialized, getClassLoader());
} catch (ClassNotFoundException e) {
throw new UtilException(e);
}
return clazz;
}
/**
* 加载类并初始化
*
* @param
* @param className 类名
* @return 类
*/
public static Class loadClass(String className) {
return loadClass(className, true);
}
/**
* 执行方法
* 可执行Private方法,也可执行static方法
* 执行非static方法时,必须满足对象有默认构造方法
* 非单例模式,如果是非静态方法,每次创建一个新对象
*
* @param
* @param classNameDotMethodName 类名和方法名表达式,例如:com.xiaoleilu.hutool.StrUtil.isEmpty
* @param args 参数,必须严格对应指定方法的参数类型和数量
* @return 返回结果
*/
public static T invoke(String classNameDotMethodName, Object... args) {
return invoke(classNameDotMethodName, false, args);
}
/**
* 执行方法
* 可执行Private方法,也可执行static方法
* 执行非static方法时,必须满足对象有默认构造方法
*
* @param
* @param classNameDotMethodName 类名和方法名表达式,例如:com.xiaoleilu.hutool.StrUtil.isEmpty
* @param isSingleton 是否为单例对象,如果此参数为false,每次执行方法时创建一个新对象
* @param args 参数,必须严格对应指定方法的参数类型和数量
* @return 返回结果
*/
public static T invoke(String classNameDotMethodName, boolean isSingleton, Object... args) {
if (StrUtil.isBlank(classNameDotMethodName)) {
throw new UtilException("Blank classNameDotMethodName!");
}
final int dotIndex = classNameDotMethodName.lastIndexOf('.');
if (dotIndex <= 0) {
throw new UtilException("Invalid classNameDotMethodName [{}]!", classNameDotMethodName);
}
final String className = classNameDotMethodName.substring(0, dotIndex);
final String methodName = classNameDotMethodName.substring(dotIndex + 1);
return invoke(className, methodName, isSingleton, args);
}
/**
* 执行方法
* 可执行Private方法,也可执行static方法
* 执行非static方法时,必须满足对象有默认构造方法
* 非单例模式,如果是非静态方法,每次创建一个新对象
*
* @param
* @param className 类名,完整类路径
* @param methodName 方法名
* @param args 参数,必须严格对应指定方法的参数类型和数量
* @return 返回结果
*/
public static T invoke(String className, String methodName, Object... args) {
return invoke(className, methodName, false, args);
}
/**
* 执行方法
* 可执行Private方法,也可执行static方法
* 执行非static方法时,必须满足对象有默认构造方法
*
* @param
* @param className 类名,完整类路径
* @param methodName 方法名
* @param isSingleton 是否为单例对象,如果此参数为false,每次执行方法时创建一个新对象
* @param args 参数,必须严格对应指定方法的参数类型和数量
* @return 返回结果
*/
public static T invoke(String className, String methodName, boolean isSingleton, Object... args) {
Class