All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.xiaoleilu.hutool.util.ClassUtil Maven / Gradle / Ivy

package com.xiaoleilu.hutool.util;

import java.io.IOException;
import java.lang.annotation.Annotation;
import java.lang.reflect.Field;
import java.lang.reflect.Method;
import java.lang.reflect.Modifier;
import java.lang.reflect.Type;
import java.net.URI;
import java.net.URL;
import java.util.ArrayList;
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 com.xiaoleilu.hutool.collection.CollectionUtil;
import com.xiaoleilu.hutool.convert.BasicType;
import com.xiaoleilu.hutool.exceptions.UtilException;
import com.xiaoleilu.hutool.io.FileUtil;
import com.xiaoleilu.hutool.io.IORuntimeException;
import com.xiaoleilu.hutool.io.resource.ResourceUtil;
import com.xiaoleilu.hutool.lang.ClassScaner;
import com.xiaoleilu.hutool.lang.Filter;
import com.xiaoleilu.hutool.lang.Singleton;

/**
 * 类工具类 
* * @author xiaoleilu * */ public class ClassUtil { /** * {@code null}安全的获取对象类型 * * @param 对象类型 * @param obj 对象,如果为{@code null} 返回{@code null} * @return 对象类型,提供对象如果为{@code null} 返回{@code null} */ @SuppressWarnings("unchecked") public static Class getClass(T obj) { return ((null == obj) ? null : (Class) obj.getClass()); } /** * 获取类名 * * @param obj 获取类名对象 * @param isSimple 是否简单类名,如果为true,返回不带包名的类名 * @return 类名 * @since 3.0.7 */ public static String getClassName(Object obj, boolean isSimple) { if (null == obj) { return null; } final Class clazz = obj.getClass(); return getClassName(clazz, isSimple); } /** * 获取类名
* 类名并不包含“.class”这个扩展名
* 例如:ClassUtil这个类
* *
	 * isSimple为false: "com.xiaoleilu.hutool.util.ClassUtil"
	 * isSimple为true: "ClassUtil"
	 * 
* * @param clazz 类 * @param isSimple 是否简单类名,如果为true,返回不带包名的类名 * @return 类名 * @since 3.0.7 */ public static String getClassName(Class clazz, boolean isSimple) { if (null == clazz) { return null; } return isSimple ? clazz.getSimpleName() : clazz.getName(); } /** * 获得对象数组的类数组 * * @param objects 对象数组,如果数组中存在{@code null}元素,则此元素被认为是Object类型 * @return 类数组 */ public static Class[] getClasses(Object... objects) { Class[] classes = new Class[objects.length]; Object obj; for (int i = 0; i < objects.length; i++) { obj = objects[i]; classes[i] = (null == obj) ? Object.class : obj.getClass(); } return classes; } /** * 指定类是否与给定的类名相同 * * @param clazz 类 * @param className 类名,可以是全类名(包含包名),也可以是简单类名(不包含包名) * @param ignoreCase 是否忽略大小写 * @return 指定类是否与给定的类名相同 * @since 3.0.7 */ public static boolean equals(Class clazz, String className, boolean ignoreCase) { if (null == clazz || StrUtil.isBlank(className)) { return false; } if (ignoreCase) { return className.equalsIgnoreCase(clazz.getName()) || className.equalsIgnoreCase(clazz.getSimpleName()); } else { return className.equals(clazz.getName()) || className.equals(clazz.getSimpleName()); } } // ----------------------------------------------------------------------------------------- Scan classes /** * 扫描指定包路径下所有包含指定注解的类 * * @param packageName 包路径 * @param annotationClass 注解类 * @return 类集合 * @see ClassScaner#scanPackageByAnnotation(String, Class) */ public static Set> scanPackageByAnnotation(String packageName, final Class annotationClass) { return ClassScaner.scanPackageByAnnotation(packageName, annotationClass); } /** * 扫描指定包路径下所有指定类或接口的子类或实现类 * * @param packageName 包路径 * @param superClass 父类或接口 * @return 类集合 * @see ClassScaner#scanPackageBySuper(String, Class) */ public static Set> scanPackageBySuper(String packageName, final Class superClass) { return ClassScaner.scanPackageBySuper(packageName, superClass); } /** * 扫面该包路径下所有class文件 * * @return 类集合 * @see ClassScaner#scanPackage() */ public static Set> scanPackage() { return ClassScaner.scanPackage(); } /** * 扫面该包路径下所有class文件 * * @param packageName 包路径 com | com. | com.abs | com.abs. * @return 类集合 * @see ClassScaner#scanPackage(String) */ public static Set> scanPackage(String packageName) { return ClassScaner.scanPackage(packageName); } /** * 扫面包路径下满足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, Filter> classFilter) { return ClassScaner.scanPackage(packageName, classFilter); } // ----------------------------------------------------------------------------------------- Method /** * 获得指定类中的Public方法名
* 去重重载的方法 * * @param clazz 类 * @return 方法名Set */ public static Set getPublicMethodNames(Class clazz) { HashSet methodSet = new HashSet(); Method[] methodArray = getPublicMethods(clazz); for (Method method : methodArray) { String methodName = method.getName(); methodSet.add(methodName); } return methodSet; } /** * 获得本类及其父类所有Public方法 * * @param clazz 查找方法的类 * @return 过滤后的方法列表 */ public static Method[] getPublicMethods(Class clazz) { return clazz.getMethods(); } /** * 获得指定类过滤后的Public方法列表 * * @param clazz 查找方法的类 * @param filter 过滤器 * @return 过滤后的方法列表 */ public static List getPublicMethods(Class clazz, Filter filter) { if (null == clazz) { return null; } Method[] methods = getPublicMethods(clazz); List methodList; if (null != filter) { methodList = new ArrayList<>(); for (Method method : methods) { if (filter.accept(method)) { methodList.add(method); } } } else { methodList = CollectionUtil.newArrayList(methods); } return methodList; } /** * 获得指定类过滤后的Public方法列表 * * @param clazz 查找方法的类 * @param excludeMethods 不包括的方法 * @return 过滤后的方法列表 */ public static List getPublicMethods(Class clazz, Method... excludeMethods) { final HashSet excludeMethodSet = CollectionUtil.newHashSet(excludeMethods); return getPublicMethods(clazz, new Filter(){ @Override public boolean accept(Method method) { return false == excludeMethodSet.contains(method); } }); } /** * 获得指定类过滤后的Public方法列表 * * @param clazz 查找方法的类 * @param excludeMethodNames 不包括的方法名列表 * @return 过滤后的方法列表 */ public static List getPublicMethods(Class clazz, String... excludeMethodNames) { final HashSet excludeMethodNameSet = CollectionUtil.newHashSet(excludeMethodNames); return getPublicMethods(clazz, new Filter(){ @Override public boolean accept(Method method) { return false == excludeMethodNameSet.contains(method.getName()); } }); } /** * 查找指定Public方法 如果找不到对应的方法或方法不为public的则返回null * * @param clazz 类 * @param methodName 方法名 * @param paramTypes 参数类型 * @return 方法 * @throws SecurityException 无权访问抛出异常 */ public static Method getPublicMethod(Class clazz, String methodName, Class... paramTypes) throws SecurityException { try { return clazz.getMethod(methodName, paramTypes); } catch (NoSuchMethodException ex) { return null; } } /** * 获得指定类中的Public方法名
* 去重重载的方法 * * @param clazz 类 * @return 方法名Set */ public static Set getDeclaredMethodNames(Class clazz) { return ReflectUtil.getMethodNames(clazz); } /** * 获得声明的所有方法,包括本类及其父类和接口的所有方法和Object类的方法 * * @param clazz 类 * @return 方法数组 */ public static Method[] getDeclaredMethods(Class clazz) { return ReflectUtil.getMethods(clazz); } /** * 查找指定对象中的所有方法(包括非public方法),也包括父对象和Object类的方法 * * @param obj 被查找的对象 * @param methodName 方法名 * @param args 参数 * @return 方法 * @throws SecurityException 无访问权限抛出异常 */ public static Method getDeclaredMethodOfObj(Object obj, String methodName, Object... args) throws SecurityException { return getDeclaredMethod(obj.getClass(), methodName, getClasses(args)); } /** * 查找指定类中的所有方法(包括非public方法),也包括父类和Object类的方法 找不到方法会返回null * * @param clazz 被查找的类 * @param methodName 方法名 * @param parameterTypes 参数类型 * @return 方法 * @throws SecurityException 无访问权限抛出异常 */ public static Method getDeclaredMethod(Class clazz, String methodName, Class... parameterTypes) throws SecurityException { return ReflectUtil.getMethod(clazz, methodName, parameterTypes); } // ----------------------------------------------------------------------------------------- Field /** * 查找指定类中的所有字段(包括非public字段), 字段不存在则返回null * * @param clazz 被查找字段的类 * @param fieldName 字段名 * @return 字段 * @throws SecurityException 安全异常 */ public static Field getDeclaredField(Class clazz, String fieldName) throws SecurityException { if (null == clazz || StrUtil.isBlank(fieldName)) { return null; } try { return clazz.getDeclaredField(fieldName); } catch (NoSuchFieldException e) { // e.printStackTrace(); } return null; } /** * 查找指定类中的所有字段(包括非public字段) * * @param clazz 被查找字段的类 * @return 字段 * @throws SecurityException 安全异常 */ public static Field[] getDeclaredFields(Class clazz) throws SecurityException { if (null == clazz) { return null; } return clazz.getDeclaredFields(); } // ----------------------------------------------------------------------------------------- Classpath /** * 获得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,将编码后的中文路径解码为原字符
* 这个ClassPath路径会文件路径被标准化处理 * * @return ClassPath */ public static String getClassPath() { return getClassPath(false); } /** * 获得ClassPath,这个ClassPath路径会文件路径被标准化处理 * * @param isEncoded 是否编码路径中的中文 * @return ClassPath * @since 3.2.1 */ public static String getClassPath(boolean isEncoded) { final URL classPathURL = getClassPathURL(); String url = isEncoded ? classPathURL.getPath() : URLUtil.getDecodedPath(classPathURL); return FileUtil.normalize(url); } /** * 获得ClassPath URL * * @return ClassPath URL */ public static URL getClassPathURL() { return getResourceURL(StrUtil.EMPTY); } /** * 获得资源的URL
* 路径用/分隔,例如: * *
	 * config/a/db.config
	 * spring/xml/test.xml
	 * 
* * @param resource 资源(相对Classpath的路径) * @return 资源URL * @see ResourceUtil#getResource(String) */ public static URL getResourceURL(String resource) throws IORuntimeException { return ResourceUtil.getResource(resource); } /** * 获取指定路径下的资源列表
* 路径格式必须为目录格式,用/分隔,例如: * *
	 * config/a
	 * spring/xml
	 * 
* * @param resource 资源路径 * @return 资源列表 * @see ResourceUtil#getResources(String) */ public static List getResources(String resource) { return ResourceUtil.getResources(resource); } /** * 获得资源相对路径对应的URL * * @param resource 资源相对路径 * @param baseClass 基准Class,获得的相对路径相对于此Class所在路径,如果为{@code null}则相对ClassPath * @return {@link URL} * @see ResourceUtil#getResource(String, Class) */ public static URL getResourceUrl(String resource, Class baseClass) { return ResourceUtil.getResource(resource, baseClass); } /** * @return 获得Java ClassPath路径,不包括 jre */ public static String[] getJavaClassPaths() { String[] classPaths = System.getProperty("java.class.path").split(System.getProperty("path.separator")); return classPaths; } /** * 获取当前线程的{@link ClassLoader} * * @return 当前线程的class loader * @see ClassLoaderUtil#getClassLoader() */ public static ClassLoader getContextClassLoader() { return ClassLoaderUtil.getContextClassLoader(); } /** * 获取{@link ClassLoader}
* 获取顺序如下:
* *
	 * 1、获取当前线程的ContextClassLoader
	 * 2、获取{@link ClassUtil}类对应的ClassLoader
	 * 3、获取系统ClassLoader({@link ClassLoader#getSystemClassLoader()})
	 * 
* * @return 类加载器 */ public static ClassLoader getClassLoader() { return ClassLoaderUtil.getClassLoader(); } /** * 比较判断types1和types2两组类,如果types1中所有的类都与types2对应位置的类相同,或者是其父类或接口,则返回true * * @param types1 类组1 * @param types2 类组2 * @return 是否相同、父类或接口 */ public static boolean isAllAssignableFrom(Class[] types1, Class[] types2) { if (ArrayUtil.isEmpty(types1) && ArrayUtil.isEmpty(types2)) { return true; } if (types1.length != types2.length) { return false; } Class type1; Class type2; for (int i = 0; i < types1.length; i++) { type1 = types1[i]; type2 = types2[i]; if(isBasicType(type1) && isBasicType(type2)) { //原始类型和包装类型存在不一致情况 if(BasicType.unWrap(type1) != BasicType.unWrap(type2)) { return false; } }else if (false == type1.isAssignableFrom(type2)) { return false; } } return true; } /** * 加载类 * * @param 对象类型 * @param className 类名 * @param isInitialized 是否初始化 * @return 类 */ @SuppressWarnings("unchecked") public static Class loadClass(String className, boolean isInitialized) { return (Class) ClassLoaderUtil.loadClass(className, isInitialized); } /** * 加载类并初始化 * * @param 对象类型 * @param className 类名 * @return 类 */ public static Class loadClass(String className) { return loadClass(className, true); } // ---------------------------------------------------------------------------------------------------- Invoke start /** * 执行方法
* 可执行Private方法,也可执行static方法
* 执行非static方法时,必须满足对象有默认构造方法
* 非单例模式,如果是非静态方法,每次创建一个新对象 * * @param 对象类型 * @param classNameDotMethodName 类名和方法名表达式,类名与方法名用.#连接 * 例如:com.xiaoleilu.hutool.StrUtil.isEmpty 或 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 classNameWithMethodName 类名和方法名表达式,例如:com.xiaoleilu.hutool.StrUtil#isEmpty或com.xiaoleilu.hutool.StrUtil.isEmpty * @param isSingleton 是否为单例对象,如果此参数为false,每次执行方法时创建一个新对象 * @param args 参数,必须严格对应指定方法的参数类型和数量 * @return 返回结果 */ public static T invoke(String classNameWithMethodName, boolean isSingleton, Object... args) { if (StrUtil.isBlank(classNameWithMethodName)) { throw new UtilException("Blank classNameDotMethodName!"); } int splitIndex = classNameWithMethodName.lastIndexOf('#'); if (splitIndex <= 0) { splitIndex = classNameWithMethodName.lastIndexOf('.'); } if (splitIndex <= 0) { throw new UtilException("Invalid classNameWithMethodName [{}]!", classNameWithMethodName); } final String className = classNameWithMethodName.substring(0, splitIndex); final String methodName = classNameWithMethodName.substring(splitIndex + 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 clazz = loadClass(className); try { final Method method = getDeclaredMethod(clazz, methodName, getClasses(args)); if (null == method) { throw new NoSuchMethodException(StrUtil.format("No such method: [{}]", methodName)); } if (isStatic(method)) { return ReflectUtil.invoke(null, method, args); } else { return ReflectUtil.invoke(isSingleton ? Singleton.get(clazz) : clazz.newInstance(), method, args); } } catch (Exception e) { throw new UtilException(e); } } // ---------------------------------------------------------------------------------------------------- Invoke end /** * 是否为包装类型 * * @param clazz 类 * @return 是否为包装类型 */ public static boolean isPrimitiveWrapper(Class clazz) { if (null == clazz) { return false; } return BasicType.wrapperPrimitiveMap.containsKey(clazz); } /** * 是否为基本类型(包括包装类和原始类) * * @param clazz 类 * @return 是否为基本类型 */ public static boolean isBasicType(Class clazz) { if (null == clazz) { return false; } return (clazz.isPrimitive() || isPrimitiveWrapper(clazz)); } /** * 是否简单值类型或简单值类型的数组
* 包括:原始类型,、String、other CharSequence, a Number, a Date, a URI, a URL, a Locale or a Class及其数组 * * @param clazz 属性类 * @return 是否简单值类型或简单值类型的数组 */ public static boolean isSimpleTypeOrArray(Class clazz) { if (null == clazz) { return false; } return isSimpleValueType(clazz) || (clazz.isArray() && isSimpleValueType(clazz.getComponentType())); } /** * 是否为简单值类型
* 包括:原始类型,、String、other CharSequence, a Number, a Date, a URI, a URL, a Locale or a Class. * * @param clazz 类 * @return 是否为简单值类型 */ public static boolean isSimpleValueType(Class clazz) { return isBasicType(clazz) || clazz.isEnum() || CharSequence.class.isAssignableFrom(clazz) || Number.class.isAssignableFrom(clazz) || Date.class.isAssignableFrom(clazz) || clazz .equals(URI.class) || clazz.equals(URL.class) || clazz.equals(Locale.class) || clazz.equals(Class.class); } /** * 检查目标类是否可以从原类转化
* 转化包括:
* 1、原类是对象,目标类型是原类型实现的接口
* 2、目标类型是原类型的父类
* 3、两者是原始类型或者包装类型(相互转换) * * @param targetType 目标类型 * @param sourceType 原类型 * @return 是否可转化 */ public static boolean isAssignable(Class targetType, Class sourceType) { if (null == targetType || null == sourceType) { return false; } // 对象类型 if (targetType.isAssignableFrom(sourceType)) { return true; } // 基本类型 if (targetType.isPrimitive()) { // 原始类型 Class resolvedPrimitive = BasicType.wrapperPrimitiveMap.get(sourceType); if (resolvedPrimitive != null && targetType.equals(resolvedPrimitive)) { return true; } } else { // 包装类型 Class resolvedWrapper = BasicType.primitiveWrapperMap.get(sourceType); if (resolvedWrapper != null && targetType.isAssignableFrom(resolvedWrapper)) { return true; } } return false; } /** * 指定类是否为Public * * @param clazz 类 * @return 是否为public */ public static boolean isPublic(Class clazz) { if (null == clazz) { throw new NullPointerException("Class to provided is null."); } return Modifier.isPublic(clazz.getModifiers()); } /** * 指定方法是否为Public * * @param method 方法 * @return 是否为public */ public static boolean isPublic(Method method) { if (null == method) { throw new NullPointerException("Method to provided is null."); } return isPublic(method.getDeclaringClass()); } /** * 指定类是否为非public * * @param clazz 类 * @return 是否为非public */ public static boolean isNotPublic(Class clazz) { return false == isPublic(clazz); } /** * 指定方法是否为非public * * @param method 方法 * @return 是否为非public */ public static boolean isNotPublic(Method method) { return false == isPublic(method); } /** * 是否为静态方法 * * @param method 方法 * @return 是否为静态方法 */ public static boolean isStatic(Method method) { return Modifier.isStatic(method.getModifiers()); } /** * 设置方法为可访问 * * @param method 方法 * @return 方法 */ public static Method setAccessible(Method method) { if (null != method && false == method.isAccessible()) { method.setAccessible(true); } return method; } /** * 是否为抽象类 * * @param clazz 类 * @return 是否为抽象类 */ public static boolean isAbstract(Class clazz) { return Modifier.isAbstract(clazz.getModifiers()); } /** * 是否为标准的类
* 这个类必须: *
	 * 1、非接口 
	 * 2、非抽象类 
	 * 3、非Enum枚举 
	 * 4、非数组 
	 * 5、非注解 
	 * 6、非原始类型(int, long等)
	 * 
* * @param clazz 类 * @return 是否为标准类 */ public static boolean isNormalClass(Class clazz) { return null != clazz // && false == clazz.isInterface() // && false == isAbstract(clazz) // && false == clazz.isEnum() // && false == clazz.isArray() // && false == clazz.isAnnotation() // && false == clazz.isSynthetic() // && false == clazz.isPrimitive();// } /** * 判断类是否为枚举类型 * @param clazz 类 * @return 是否为枚举类型 * @since 3.2.0 */ public static boolean isEnum(Class clazz) { return null == clazz ? false : clazz.isEnum(); } /** * 获得给定类的第一个泛型参数 * * @param clazz 被检查的类,必须是已经确定泛型类型的类 * @return {@link Class} */ public static Class getTypeArgument(Class clazz) { return getTypeArgument(clazz, 0); } /** * 获得给定类的泛型参数 * * @param clazz 被检查的类,必须是已经确定泛型类型的类 * @param index 泛型类型的索引号,既第几个泛型类型 * @return {@link Class} */ public static Class getTypeArgument(Class clazz, int index) { final Type argumentType = TypeUtil.getTypeArgument(clazz, index); if (null != argumentType && argumentType instanceof Class) { return (Class) argumentType; } return null; } /** * 获得给定类所在包的名称
* 例如:
* com.xiaoleilu.hutool.util.ClassUtil =》 com.xiaoleilu.hutool.util * * @param clazz 类 * @return 包名 */ public static String getPackage(Class clazz) { if (clazz == null) { return StrUtil.EMPTY; } final String className = clazz.getName(); int packageEndIndex = className.lastIndexOf(StrUtil.DOT); if (packageEndIndex == -1) { return StrUtil.EMPTY; } return className.substring(0, packageEndIndex); } /** * 获得给定类所在包的路径
* 例如:
* com.xiaoleilu.hutool.util.ClassUtil =》 com/xiaoleilu/hutool/util * * @param clazz 类 * @return 包名 */ public static String getPackagePath(Class clazz) { return getPackage(clazz).replace(StrUtil.C_DOT, StrUtil.C_SLASH); } /** * 获取指定类型分的默认值
* 默认值规则为: *
	 * 1、如果为原始类型,返回0
	 * 2、非原始类型返回{@code null}
	 * 
* * @param clazz 类 * @return 默认值 * @since 3.0.8 */ public static Object getDefaultValue(Class clazz) { if(clazz.isPrimitive()) { if(long.class == clazz) { return 0L; }else if(int.class == clazz) { return 0; }else if(short.class == clazz) { return (short)0; }else if(char.class == clazz) { return (char)0; }else if(byte.class == clazz) { return (byte)0; }else if(double.class == clazz) { return 0D; }else if(float.class == clazz) { return 0f; }else if(boolean.class == clazz) { return false; } } return null; } /** * 获得默认值列表 * @param classes 值类型 * @return 默认值列表 * @since 3.0.9 */ public static Object[] getDefaultValues(Class... classes) { final Object[] values = new Object[classes.length]; for(int i = 0; i < classes.length; i++) { values[i] = getDefaultValue(classes[i]); } return values; } }