ÎļþÃû´Ó ruoyi/src/main/java/com/ruoyi/common/utils/reflect/ReflectUtils.java ÐÞ¸Ä |
| | |
| | | package com.ruoyi.common.utils.reflect; |
| | | |
| | | import java.lang.reflect.Field; |
| | | import java.lang.reflect.InvocationTargetException; |
| | | import java.lang.reflect.Method; |
| | | import java.lang.reflect.Modifier; |
| | | import java.lang.reflect.ParameterizedType; |
| | | import java.lang.reflect.Type; |
| | | import java.util.Date; |
| | | import org.apache.commons.lang3.StringUtils; |
| | | import org.apache.commons.lang3.Validate; |
| | | import org.apache.poi.ss.usermodel.DateUtil; |
| | | import org.slf4j.Logger; |
| | | import org.slf4j.LoggerFactory; |
| | | import com.ruoyi.common.core.text.Convert; |
| | | import com.ruoyi.common.utils.DateUtils; |
| | | |
| | | /** |
| | | * åå°å·¥å
·ç±». æä¾è°ç¨getter/setteræ¹æ³, 访é®ç§æåé, è°ç¨ç§ææ¹æ³, è·åæ³åç±»åClass, 被AOPè¿ççå®ç±»çå·¥å
·å½æ°. |
| | | * |
| | | * @author ruoyi |
| | | */ |
| | | @SuppressWarnings("rawtypes") |
| | | public class ReflectUtils |
| | | { |
| | | private static final String SETTER_PREFIX = "set"; |
| | | |
| | | private static final String GETTER_PREFIX = "get"; |
| | | |
| | | private static final String CGLIB_CLASS_SEPARATOR = "$$"; |
| | | |
| | | private static Logger logger = LoggerFactory.getLogger(ReflectUtils.class); |
| | | |
| | | /** |
| | | * è°ç¨Getteræ¹æ³. |
| | | * æ¯æå¤çº§ï¼å¦ï¼å¯¹è±¡å.对象å.æ¹æ³ |
| | | */ |
| | | @SuppressWarnings("unchecked") |
| | | public static <E> E invokeGetter(Object obj, String propertyName) |
| | | { |
| | | Object object = obj; |
| | | for (String name : StringUtils.split(propertyName, ".")) |
| | | { |
| | | String getterMethodName = GETTER_PREFIX + StringUtils.capitalize(name); |
| | | object = invokeMethod(object, getterMethodName, new Class[] {}, new Object[] {}); |
| | | } |
| | | return (E) object; |
| | | } |
| | | |
| | | /** |
| | | * è°ç¨Setteræ¹æ³, ä»
å¹é
æ¹æ³åã |
| | | * æ¯æå¤çº§ï¼å¦ï¼å¯¹è±¡å.对象å.æ¹æ³ |
| | | */ |
| | | public static <E> void invokeSetter(Object obj, String propertyName, E value) |
| | | { |
| | | Object object = obj; |
| | | String[] names = StringUtils.split(propertyName, "."); |
| | | for (int i = 0; i < names.length; i++) |
| | | { |
| | | if (i < names.length - 1) |
| | | { |
| | | String getterMethodName = GETTER_PREFIX + StringUtils.capitalize(names[i]); |
| | | object = invokeMethod(object, getterMethodName, new Class[] {}, new Object[] {}); |
| | | } |
| | | else |
| | | { |
| | | String setterMethodName = SETTER_PREFIX + StringUtils.capitalize(names[i]); |
| | | invokeMethodByName(object, setterMethodName, new Object[] { value }); |
| | | } |
| | | } |
| | | } |
| | | |
| | | /** |
| | | * ç´æ¥è¯»åå¯¹è±¡å±æ§å¼, æ è§private/protected修饰符, ä¸ç»è¿getter彿°. |
| | | */ |
| | | @SuppressWarnings("unchecked") |
| | | public static <E> E getFieldValue(final Object obj, final String fieldName) |
| | | { |
| | | Field field = getAccessibleField(obj, fieldName); |
| | | if (field == null) |
| | | { |
| | | logger.debug("å¨ [" + obj.getClass() + "] ä¸ï¼æ²¡ææ¾å° [" + fieldName + "] åæ®µ "); |
| | | return null; |
| | | } |
| | | E result = null; |
| | | try |
| | | { |
| | | result = (E) field.get(obj); |
| | | } |
| | | catch (IllegalAccessException e) |
| | | { |
| | | logger.error("ä¸å¯è½æåºçå¼å¸¸{}", e.getMessage()); |
| | | } |
| | | return result; |
| | | } |
| | | |
| | | /** |
| | | * ç´æ¥è®¾ç½®å¯¹è±¡å±æ§å¼, æ è§private/protected修饰符, ä¸ç»è¿setter彿°. |
| | | */ |
| | | public static <E> void setFieldValue(final Object obj, final String fieldName, final E value) |
| | | { |
| | | Field field = getAccessibleField(obj, fieldName); |
| | | if (field == null) |
| | | { |
| | | // throw new IllegalArgumentException("å¨ [" + obj.getClass() + "] ä¸ï¼æ²¡ææ¾å° [" + fieldName + "] åæ®µ "); |
| | | logger.debug("å¨ [" + obj.getClass() + "] ä¸ï¼æ²¡ææ¾å° [" + fieldName + "] åæ®µ "); |
| | | return; |
| | | } |
| | | try |
| | | { |
| | | field.set(obj, value); |
| | | } |
| | | catch (IllegalAccessException e) |
| | | { |
| | | logger.error("ä¸å¯è½æåºçå¼å¸¸: {}", e.getMessage()); |
| | | } |
| | | } |
| | | |
| | | /** |
| | | * ç´æ¥è°ç¨å¯¹è±¡æ¹æ³, æ è§private/protected修饰符. |
| | | * ç¨äºä¸æ¬¡æ§è°ç¨çæ
åµï¼å¦ååºä½¿ç¨getAccessibleMethod()彿°è·å¾Methodååå¤è°ç¨. |
| | | * åæ¶å¹é
æ¹æ³å+åæ°ç±»åï¼ |
| | | */ |
| | | @SuppressWarnings("unchecked") |
| | | public static <E> E invokeMethod(final Object obj, final String methodName, final Class<?>[] parameterTypes, |
| | | final Object[] args) |
| | | { |
| | | if (obj == null || methodName == null) |
| | | { |
| | | return null; |
| | | } |
| | | Method method = getAccessibleMethod(obj, methodName, parameterTypes); |
| | | if (method == null) |
| | | { |
| | | logger.debug("å¨ [" + obj.getClass() + "] ä¸ï¼æ²¡ææ¾å° [" + methodName + "] æ¹æ³ "); |
| | | return null; |
| | | } |
| | | try |
| | | { |
| | | return (E) method.invoke(obj, args); |
| | | } |
| | | catch (Exception e) |
| | | { |
| | | String msg = "method: " + method + ", obj: " + obj + ", args: " + args + ""; |
| | | throw convertReflectionExceptionToUnchecked(msg, e); |
| | | } |
| | | } |
| | | |
| | | /** |
| | | * ç´æ¥è°ç¨å¯¹è±¡æ¹æ³, æ è§private/protectedä¿®é¥°ç¬¦ï¼ |
| | | * ç¨äºä¸æ¬¡æ§è°ç¨çæ
åµï¼å¦ååºä½¿ç¨getAccessibleMethodByName()彿°è·å¾Methodååå¤è°ç¨. |
| | | * åªå¹é
彿°åï¼å¦ææå¤ä¸ªåå彿°è°ç¨ç¬¬ä¸ä¸ªã |
| | | */ |
| | | @SuppressWarnings("unchecked") |
| | | public static <E> E invokeMethodByName(final Object obj, final String methodName, final Object[] args) |
| | | { |
| | | Method method = getAccessibleMethodByName(obj, methodName, args.length); |
| | | if (method == null) |
| | | { |
| | | // å¦æä¸ºç©ºä¸æ¥éï¼ç´æ¥è¿å空ã |
| | | logger.debug("å¨ [" + obj.getClass() + "] ä¸ï¼æ²¡ææ¾å° [" + methodName + "] æ¹æ³ "); |
| | | return null; |
| | | } |
| | | try |
| | | { |
| | | // ç±»å转æ¢ï¼å°åæ°æ°æ®ç±»å转æ¢ä¸ºç®æ æ¹æ³åæ°ç±»åï¼ |
| | | Class<?>[] cs = method.getParameterTypes(); |
| | | for (int i = 0; i < cs.length; i++) |
| | | { |
| | | if (args[i] != null && !args[i].getClass().equals(cs[i])) |
| | | { |
| | | if (cs[i] == String.class) |
| | | { |
| | | args[i] = Convert.toStr(args[i]); |
| | | if (StringUtils.endsWith((String) args[i], ".0")) |
| | | { |
| | | args[i] = StringUtils.substringBefore((String) args[i], ".0"); |
| | | } |
| | | } |
| | | else if (cs[i] == Integer.class) |
| | | { |
| | | args[i] = Convert.toInt(args[i]); |
| | | } |
| | | else if (cs[i] == Long.class) |
| | | { |
| | | args[i] = Convert.toLong(args[i]); |
| | | } |
| | | else if (cs[i] == Double.class) |
| | | { |
| | | args[i] = Convert.toDouble(args[i]); |
| | | } |
| | | else if (cs[i] == Float.class) |
| | | { |
| | | args[i] = Convert.toFloat(args[i]); |
| | | } |
| | | else if (cs[i] == Date.class) |
| | | { |
| | | if (args[i] instanceof String) |
| | | { |
| | | args[i] = DateUtils.parseDate(args[i]); |
| | | } |
| | | else |
| | | { |
| | | args[i] = DateUtil.getJavaDate((Double) args[i]); |
| | | } |
| | | } |
| | | } |
| | | } |
| | | return (E) method.invoke(obj, args); |
| | | } |
| | | catch (Exception e) |
| | | { |
| | | String msg = "method: " + method + ", obj: " + obj + ", args: " + args + ""; |
| | | throw convertReflectionExceptionToUnchecked(msg, e); |
| | | } |
| | | } |
| | | |
| | | /** |
| | | * 循ç¯åä¸è½¬å, è·å对象çDeclaredField, 并强å¶è®¾ç½®ä¸ºå¯è®¿é®. |
| | | * å¦åä¸è½¬åå°Object仿 æ³æ¾å°, è¿ånull. |
| | | */ |
| | | public static Field getAccessibleField(final Object obj, final String fieldName) |
| | | { |
| | | // ä¸ºç©ºä¸æ¥éãç´æ¥è¿å null |
| | | if (obj == null) |
| | | { |
| | | return null; |
| | | } |
| | | Validate.notBlank(fieldName, "fieldName can't be blank"); |
| | | for (Class<?> superClass = obj.getClass(); superClass != Object.class; superClass = superClass.getSuperclass()) |
| | | { |
| | | try |
| | | { |
| | | Field field = superClass.getDeclaredField(fieldName); |
| | | makeAccessible(field); |
| | | return field; |
| | | } |
| | | catch (NoSuchFieldException e) |
| | | { |
| | | continue; |
| | | } |
| | | } |
| | | return null; |
| | | } |
| | | |
| | | /** |
| | | * 循ç¯åä¸è½¬å, è·å对象çDeclaredMethod,并强å¶è®¾ç½®ä¸ºå¯è®¿é®. |
| | | * å¦åä¸è½¬åå°Object仿 æ³æ¾å°, è¿ånull. |
| | | * å¹é
彿°å+åæ°ç±»åã |
| | | * ç¨äºæ¹æ³éè¦è¢«å¤æ¬¡è°ç¨çæ
åµ. å
ä½¿ç¨æ¬å½æ°å
åå¾Method,ç¶åè°ç¨Method.invoke(Object obj, Object... args) |
| | | */ |
| | | public static Method getAccessibleMethod(final Object obj, final String methodName, |
| | | final Class<?>... parameterTypes) |
| | | { |
| | | // ä¸ºç©ºä¸æ¥éãç´æ¥è¿å null |
| | | if (obj == null) |
| | | { |
| | | return null; |
| | | } |
| | | Validate.notBlank(methodName, "methodName can't be blank"); |
| | | for (Class<?> searchType = obj.getClass(); searchType != Object.class; searchType = searchType.getSuperclass()) |
| | | { |
| | | try |
| | | { |
| | | Method method = searchType.getDeclaredMethod(methodName, parameterTypes); |
| | | makeAccessible(method); |
| | | return method; |
| | | } |
| | | catch (NoSuchMethodException e) |
| | | { |
| | | continue; |
| | | } |
| | | } |
| | | return null; |
| | | } |
| | | |
| | | /** |
| | | * 循ç¯åä¸è½¬å, è·å对象çDeclaredMethod,并强å¶è®¾ç½®ä¸ºå¯è®¿é®. |
| | | * å¦åä¸è½¬åå°Object仿 æ³æ¾å°, è¿ånull. |
| | | * åªå¹é
彿°åã |
| | | * ç¨äºæ¹æ³éè¦è¢«å¤æ¬¡è°ç¨çæ
åµ. å
ä½¿ç¨æ¬å½æ°å
åå¾Method,ç¶åè°ç¨Method.invoke(Object obj, Object... args) |
| | | */ |
| | | public static Method getAccessibleMethodByName(final Object obj, final String methodName, int argsNum) |
| | | { |
| | | // ä¸ºç©ºä¸æ¥éãç´æ¥è¿å null |
| | | if (obj == null) |
| | | { |
| | | return null; |
| | | } |
| | | Validate.notBlank(methodName, "methodName can't be blank"); |
| | | for (Class<?> searchType = obj.getClass(); searchType != Object.class; searchType = searchType.getSuperclass()) |
| | | { |
| | | Method[] methods = searchType.getDeclaredMethods(); |
| | | for (Method method : methods) |
| | | { |
| | | if (method.getName().equals(methodName) && method.getParameterTypes().length == argsNum) |
| | | { |
| | | makeAccessible(method); |
| | | return method; |
| | | } |
| | | } |
| | | } |
| | | return null; |
| | | } |
| | | |
| | | /** |
| | | * æ¹åprivate/protectedçæ¹æ³ä¸ºpublicï¼å°½éä¸è°ç¨å®é
æ¹å¨çè¯å¥ï¼é¿å
JDKçSecurityManageræ±æ¨ã |
| | | */ |
| | | public static void makeAccessible(Method method) |
| | | { |
| | | if ((!Modifier.isPublic(method.getModifiers()) || !Modifier.isPublic(method.getDeclaringClass().getModifiers())) |
| | | && !method.isAccessible()) |
| | | { |
| | | method.setAccessible(true); |
| | | } |
| | | } |
| | | |
| | | /** |
| | | * æ¹åprivate/protectedçæååé为publicï¼å°½éä¸è°ç¨å®é
æ¹å¨çè¯å¥ï¼é¿å
JDKçSecurityManageræ±æ¨ã |
| | | */ |
| | | public static void makeAccessible(Field field) |
| | | { |
| | | if ((!Modifier.isPublic(field.getModifiers()) || !Modifier.isPublic(field.getDeclaringClass().getModifiers()) |
| | | || Modifier.isFinal(field.getModifiers())) && !field.isAccessible()) |
| | | { |
| | | field.setAccessible(true); |
| | | } |
| | | } |
| | | |
| | | /** |
| | | * éè¿åå°, è·å¾Classå®ä¹ä¸å£°æçæ³ååæ°çç±»å, æ³¨ææ³åå¿
é¡»å®ä¹å¨ç¶ç±»å¤ |
| | | * 妿 æ³æ¾å°, è¿åObject.class. |
| | | */ |
| | | @SuppressWarnings("unchecked") |
| | | public static <T> Class<T> getClassGenricType(final Class clazz) |
| | | { |
| | | return getClassGenricType(clazz, 0); |
| | | } |
| | | |
| | | /** |
| | | * éè¿åå°, è·å¾Classå®ä¹ä¸å£°æçç¶ç±»çæ³ååæ°çç±»å. |
| | | * 妿 æ³æ¾å°, è¿åObject.class. |
| | | */ |
| | | public static Class getClassGenricType(final Class clazz, final int index) |
| | | { |
| | | Type genType = clazz.getGenericSuperclass(); |
| | | |
| | | if (!(genType instanceof ParameterizedType)) |
| | | { |
| | | logger.debug(clazz.getSimpleName() + "'s superclass not ParameterizedType"); |
| | | return Object.class; |
| | | } |
| | | |
| | | Type[] params = ((ParameterizedType) genType).getActualTypeArguments(); |
| | | |
| | | if (index >= params.length || index < 0) |
| | | { |
| | | logger.debug("Index: " + index + ", Size of " + clazz.getSimpleName() + "'s Parameterized Type: " |
| | | + params.length); |
| | | return Object.class; |
| | | } |
| | | if (!(params[index] instanceof Class)) |
| | | { |
| | | logger.debug(clazz.getSimpleName() + " not set the actual class on superclass generic parameter"); |
| | | return Object.class; |
| | | } |
| | | |
| | | return (Class) params[index]; |
| | | } |
| | | |
| | | public static Class<?> getUserClass(Object instance) |
| | | { |
| | | if (instance == null) |
| | | { |
| | | throw new RuntimeException("Instance must not be null"); |
| | | } |
| | | Class clazz = instance.getClass(); |
| | | if (clazz != null && clazz.getName().contains(CGLIB_CLASS_SEPARATOR)) |
| | | { |
| | | Class<?> superClass = clazz.getSuperclass(); |
| | | if (superClass != null && !Object.class.equals(superClass)) |
| | | { |
| | | return superClass; |
| | | } |
| | | } |
| | | return clazz; |
| | | |
| | | } |
| | | |
| | | /** |
| | | * å°åå°æ¶çchecked exception转æ¢ä¸ºunchecked exception. |
| | | */ |
| | | public static RuntimeException convertReflectionExceptionToUnchecked(String msg, Exception e) |
| | | { |
| | | if (e instanceof IllegalAccessException || e instanceof IllegalArgumentException |
| | | || e instanceof NoSuchMethodException) |
| | | { |
| | | return new IllegalArgumentException(msg, e); |
| | | } |
| | | else if (e instanceof InvocationTargetException) |
| | | { |
| | | return new RuntimeException(msg, ((InvocationTargetException) e).getTargetException()); |
| | | } |
| | | return new RuntimeException(msg, e); |
| | | } |
| | | } |
| | | package com.ruoyi.common.utils.reflect;
|
| | |
|
| | | import java.lang.reflect.Field;
|
| | | import java.lang.reflect.InvocationTargetException;
|
| | | import java.lang.reflect.Method;
|
| | | import java.lang.reflect.Modifier;
|
| | | import java.lang.reflect.ParameterizedType;
|
| | | import java.lang.reflect.Type;
|
| | | import java.util.Date;
|
| | | import org.apache.commons.lang3.StringUtils;
|
| | | import org.apache.commons.lang3.Validate;
|
| | | import org.apache.poi.ss.usermodel.DateUtil;
|
| | | import org.slf4j.Logger;
|
| | | import org.slf4j.LoggerFactory;
|
| | | import com.ruoyi.common.core.text.Convert;
|
| | | import com.ruoyi.common.utils.DateUtils;
|
| | |
|
| | | /**
|
| | | * åå°å·¥å
·ç±». æä¾è°ç¨getter/setteræ¹æ³, 访é®ç§æåé, è°ç¨ç§ææ¹æ³, è·åæ³åç±»åClass, 被AOPè¿ççå®ç±»çå·¥å
·å½æ°.
|
| | | * |
| | | * @author ruoyi
|
| | | */
|
| | | @SuppressWarnings("rawtypes")
|
| | | public class ReflectUtils
|
| | | {
|
| | | private static final String SETTER_PREFIX = "set";
|
| | |
|
| | | private static final String GETTER_PREFIX = "get";
|
| | |
|
| | | private static final String CGLIB_CLASS_SEPARATOR = "$$";
|
| | |
|
| | | private static Logger logger = LoggerFactory.getLogger(ReflectUtils.class);
|
| | |
|
| | | /**
|
| | | * è°ç¨Getteræ¹æ³.
|
| | | * æ¯æå¤çº§ï¼å¦ï¼å¯¹è±¡å.对象å.æ¹æ³
|
| | | */
|
| | | @SuppressWarnings("unchecked")
|
| | | public static <E> E invokeGetter(Object obj, String propertyName)
|
| | | {
|
| | | Object object = obj;
|
| | | for (String name : StringUtils.split(propertyName, "."))
|
| | | {
|
| | | String getterMethodName = GETTER_PREFIX + StringUtils.capitalize(name);
|
| | | object = invokeMethod(object, getterMethodName, new Class[] {}, new Object[] {});
|
| | | }
|
| | | return (E) object;
|
| | | }
|
| | |
|
| | | /**
|
| | | * è°ç¨Setteræ¹æ³, ä»
å¹é
æ¹æ³åã
|
| | | * æ¯æå¤çº§ï¼å¦ï¼å¯¹è±¡å.对象å.æ¹æ³
|
| | | */
|
| | | public static <E> void invokeSetter(Object obj, String propertyName, E value)
|
| | | {
|
| | | Object object = obj;
|
| | | String[] names = StringUtils.split(propertyName, ".");
|
| | | for (int i = 0; i < names.length; i++)
|
| | | {
|
| | | if (i < names.length - 1)
|
| | | {
|
| | | String getterMethodName = GETTER_PREFIX + StringUtils.capitalize(names[i]);
|
| | | object = invokeMethod(object, getterMethodName, new Class[] {}, new Object[] {});
|
| | | }
|
| | | else
|
| | | {
|
| | | String setterMethodName = SETTER_PREFIX + StringUtils.capitalize(names[i]);
|
| | | invokeMethodByName(object, setterMethodName, new Object[] { value });
|
| | | }
|
| | | }
|
| | | }
|
| | |
|
| | | /**
|
| | | * ç´æ¥è¯»åå¯¹è±¡å±æ§å¼, æ è§private/protected修饰符, ä¸ç»è¿getter彿°.
|
| | | */
|
| | | @SuppressWarnings("unchecked")
|
| | | public static <E> E getFieldValue(final Object obj, final String fieldName)
|
| | | {
|
| | | Field field = getAccessibleField(obj, fieldName);
|
| | | if (field == null)
|
| | | {
|
| | | logger.debug("å¨ [" + obj.getClass() + "] ä¸ï¼æ²¡ææ¾å° [" + fieldName + "] åæ®µ ");
|
| | | return null;
|
| | | }
|
| | | E result = null;
|
| | | try
|
| | | {
|
| | | result = (E) field.get(obj);
|
| | | }
|
| | | catch (IllegalAccessException e)
|
| | | {
|
| | | logger.error("ä¸å¯è½æåºçå¼å¸¸{}", e.getMessage());
|
| | | }
|
| | | return result;
|
| | | }
|
| | |
|
| | | /**
|
| | | * ç´æ¥è®¾ç½®å¯¹è±¡å±æ§å¼, æ è§private/protected修饰符, ä¸ç»è¿setter彿°.
|
| | | */
|
| | | public static <E> void setFieldValue(final Object obj, final String fieldName, final E value)
|
| | | {
|
| | | Field field = getAccessibleField(obj, fieldName);
|
| | | if (field == null)
|
| | | {
|
| | | // throw new IllegalArgumentException("å¨ [" + obj.getClass() + "] ä¸ï¼æ²¡ææ¾å° [" + fieldName + "] åæ®µ ");
|
| | | logger.debug("å¨ [" + obj.getClass() + "] ä¸ï¼æ²¡ææ¾å° [" + fieldName + "] åæ®µ ");
|
| | | return;
|
| | | }
|
| | | try
|
| | | {
|
| | | field.set(obj, value);
|
| | | }
|
| | | catch (IllegalAccessException e)
|
| | | {
|
| | | logger.error("ä¸å¯è½æåºçå¼å¸¸: {}", e.getMessage());
|
| | | }
|
| | | }
|
| | |
|
| | | /**
|
| | | * ç´æ¥è°ç¨å¯¹è±¡æ¹æ³, æ è§private/protected修饰符.
|
| | | * ç¨äºä¸æ¬¡æ§è°ç¨çæ
åµï¼å¦ååºä½¿ç¨getAccessibleMethod()彿°è·å¾Methodååå¤è°ç¨.
|
| | | * åæ¶å¹é
æ¹æ³å+åæ°ç±»åï¼
|
| | | */
|
| | | @SuppressWarnings("unchecked")
|
| | | public static <E> E invokeMethod(final Object obj, final String methodName, final Class<?>[] parameterTypes,
|
| | | final Object[] args)
|
| | | {
|
| | | if (obj == null || methodName == null)
|
| | | {
|
| | | return null;
|
| | | }
|
| | | Method method = getAccessibleMethod(obj, methodName, parameterTypes);
|
| | | if (method == null)
|
| | | {
|
| | | logger.debug("å¨ [" + obj.getClass() + "] ä¸ï¼æ²¡ææ¾å° [" + methodName + "] æ¹æ³ ");
|
| | | return null;
|
| | | }
|
| | | try
|
| | | {
|
| | | return (E) method.invoke(obj, args);
|
| | | }
|
| | | catch (Exception e)
|
| | | {
|
| | | String msg = "method: " + method + ", obj: " + obj + ", args: " + args + "";
|
| | | throw convertReflectionExceptionToUnchecked(msg, e);
|
| | | }
|
| | | }
|
| | |
|
| | | /**
|
| | | * ç´æ¥è°ç¨å¯¹è±¡æ¹æ³, æ è§private/protected修饰符ï¼
|
| | | * ç¨äºä¸æ¬¡æ§è°ç¨çæ
åµï¼å¦ååºä½¿ç¨getAccessibleMethodByName()彿°è·å¾Methodååå¤è°ç¨.
|
| | | * åªå¹é
彿°åï¼å¦ææå¤ä¸ªåå彿°è°ç¨ç¬¬ä¸ä¸ªã
|
| | | */
|
| | | @SuppressWarnings("unchecked")
|
| | | public static <E> E invokeMethodByName(final Object obj, final String methodName, final Object[] args)
|
| | | {
|
| | | Method method = getAccessibleMethodByName(obj, methodName, args.length);
|
| | | if (method == null)
|
| | | {
|
| | | // å¦æä¸ºç©ºä¸æ¥éï¼ç´æ¥è¿å空ã
|
| | | logger.debug("å¨ [" + obj.getClass() + "] ä¸ï¼æ²¡ææ¾å° [" + methodName + "] æ¹æ³ ");
|
| | | return null;
|
| | | }
|
| | | try
|
| | | {
|
| | | // ç±»å转æ¢ï¼å°åæ°æ°æ®ç±»å转æ¢ä¸ºç®æ æ¹æ³åæ°ç±»åï¼
|
| | | Class<?>[] cs = method.getParameterTypes();
|
| | | for (int i = 0; i < cs.length; i++)
|
| | | {
|
| | | if (args[i] != null && !args[i].getClass().equals(cs[i]))
|
| | | {
|
| | | if (cs[i] == String.class)
|
| | | {
|
| | | args[i] = Convert.toStr(args[i]);
|
| | | if (StringUtils.endsWith((String) args[i], ".0"))
|
| | | {
|
| | | args[i] = StringUtils.substringBefore((String) args[i], ".0");
|
| | | }
|
| | | }
|
| | | else if (cs[i] == Integer.class)
|
| | | {
|
| | | args[i] = Convert.toInt(args[i]);
|
| | | }
|
| | | else if (cs[i] == Long.class)
|
| | | {
|
| | | args[i] = Convert.toLong(args[i]);
|
| | | }
|
| | | else if (cs[i] == Double.class)
|
| | | {
|
| | | args[i] = Convert.toDouble(args[i]);
|
| | | }
|
| | | else if (cs[i] == Float.class)
|
| | | {
|
| | | args[i] = Convert.toFloat(args[i]);
|
| | | }
|
| | | else if (cs[i] == Date.class)
|
| | | {
|
| | | if (args[i] instanceof String)
|
| | | {
|
| | | args[i] = DateUtils.parseDate(args[i]);
|
| | | }
|
| | | else
|
| | | {
|
| | | args[i] = DateUtil.getJavaDate((Double) args[i]);
|
| | | }
|
| | | }
|
| | | }
|
| | | }
|
| | | return (E) method.invoke(obj, args);
|
| | | }
|
| | | catch (Exception e)
|
| | | {
|
| | | String msg = "method: " + method + ", obj: " + obj + ", args: " + args + "";
|
| | | throw convertReflectionExceptionToUnchecked(msg, e);
|
| | | }
|
| | | }
|
| | |
|
| | | /**
|
| | | * 循ç¯åä¸è½¬å, è·å对象çDeclaredField, 并强å¶è®¾ç½®ä¸ºå¯è®¿é®.
|
| | | * å¦åä¸è½¬åå°Object仿 æ³æ¾å°, è¿ånull.
|
| | | */
|
| | | public static Field getAccessibleField(final Object obj, final String fieldName)
|
| | | {
|
| | | // ä¸ºç©ºä¸æ¥éãç´æ¥è¿å null
|
| | | if (obj == null)
|
| | | {
|
| | | return null;
|
| | | }
|
| | | Validate.notBlank(fieldName, "fieldName can't be blank");
|
| | | for (Class<?> superClass = obj.getClass(); superClass != Object.class; superClass = superClass.getSuperclass())
|
| | | {
|
| | | try
|
| | | {
|
| | | Field field = superClass.getDeclaredField(fieldName);
|
| | | makeAccessible(field);
|
| | | return field;
|
| | | }
|
| | | catch (NoSuchFieldException e)
|
| | | {
|
| | | continue;
|
| | | }
|
| | | }
|
| | | return null;
|
| | | }
|
| | |
|
| | | /**
|
| | | * 循ç¯åä¸è½¬å, è·å对象çDeclaredMethod,并强å¶è®¾ç½®ä¸ºå¯è®¿é®.
|
| | | * å¦åä¸è½¬åå°Object仿 æ³æ¾å°, è¿ånull.
|
| | | * å¹é
彿°å+åæ°ç±»åã
|
| | | * ç¨äºæ¹æ³éè¦è¢«å¤æ¬¡è°ç¨çæ
åµ. å
ä½¿ç¨æ¬å½æ°å
åå¾Method,ç¶åè°ç¨Method.invoke(Object obj, Object... args)
|
| | | */
|
| | | public static Method getAccessibleMethod(final Object obj, final String methodName,
|
| | | final Class<?>... parameterTypes)
|
| | | {
|
| | | // ä¸ºç©ºä¸æ¥éãç´æ¥è¿å null
|
| | | if (obj == null)
|
| | | {
|
| | | return null;
|
| | | }
|
| | | Validate.notBlank(methodName, "methodName can't be blank");
|
| | | for (Class<?> searchType = obj.getClass(); searchType != Object.class; searchType = searchType.getSuperclass())
|
| | | {
|
| | | try
|
| | | {
|
| | | Method method = searchType.getDeclaredMethod(methodName, parameterTypes);
|
| | | makeAccessible(method);
|
| | | return method;
|
| | | }
|
| | | catch (NoSuchMethodException e)
|
| | | {
|
| | | continue;
|
| | | }
|
| | | }
|
| | | return null;
|
| | | }
|
| | |
|
| | | /**
|
| | | * 循ç¯åä¸è½¬å, è·å对象çDeclaredMethod,并强å¶è®¾ç½®ä¸ºå¯è®¿é®.
|
| | | * å¦åä¸è½¬åå°Object仿 æ³æ¾å°, è¿ånull.
|
| | | * åªå¹é
彿°åã
|
| | | * ç¨äºæ¹æ³éè¦è¢«å¤æ¬¡è°ç¨çæ
åµ. å
ä½¿ç¨æ¬å½æ°å
åå¾Method,ç¶åè°ç¨Method.invoke(Object obj, Object... args)
|
| | | */
|
| | | public static Method getAccessibleMethodByName(final Object obj, final String methodName, int argsNum)
|
| | | {
|
| | | // ä¸ºç©ºä¸æ¥éãç´æ¥è¿å null
|
| | | if (obj == null)
|
| | | {
|
| | | return null;
|
| | | }
|
| | | Validate.notBlank(methodName, "methodName can't be blank");
|
| | | for (Class<?> searchType = obj.getClass(); searchType != Object.class; searchType = searchType.getSuperclass())
|
| | | {
|
| | | Method[] methods = searchType.getDeclaredMethods();
|
| | | for (Method method : methods)
|
| | | {
|
| | | if (method.getName().equals(methodName) && method.getParameterTypes().length == argsNum)
|
| | | {
|
| | | makeAccessible(method);
|
| | | return method;
|
| | | }
|
| | | }
|
| | | }
|
| | | return null;
|
| | | }
|
| | |
|
| | | /**
|
| | | * æ¹åprivate/protectedçæ¹æ³ä¸ºpublicï¼å°½éä¸è°ç¨å®é
æ¹å¨çè¯å¥ï¼é¿å
JDKçSecurityManageræ±æ¨ã
|
| | | */
|
| | | public static void makeAccessible(Method method)
|
| | | {
|
| | | if ((!Modifier.isPublic(method.getModifiers()) || !Modifier.isPublic(method.getDeclaringClass().getModifiers()))
|
| | | && !method.isAccessible())
|
| | | {
|
| | | method.setAccessible(true);
|
| | | }
|
| | | }
|
| | |
|
| | | /**
|
| | | * æ¹åprivate/protectedçæååé为publicï¼å°½éä¸è°ç¨å®é
æ¹å¨çè¯å¥ï¼é¿å
JDKçSecurityManageræ±æ¨ã
|
| | | */
|
| | | public static void makeAccessible(Field field)
|
| | | {
|
| | | if ((!Modifier.isPublic(field.getModifiers()) || !Modifier.isPublic(field.getDeclaringClass().getModifiers())
|
| | | || Modifier.isFinal(field.getModifiers())) && !field.isAccessible())
|
| | | {
|
| | | field.setAccessible(true);
|
| | | }
|
| | | }
|
| | |
|
| | | /**
|
| | | * éè¿åå°, è·å¾Classå®ä¹ä¸å£°æçæ³ååæ°çç±»å, æ³¨ææ³åå¿
é¡»å®ä¹å¨ç¶ç±»å¤
|
| | | * 妿 æ³æ¾å°, è¿åObject.class.
|
| | | */
|
| | | @SuppressWarnings("unchecked")
|
| | | public static <T> Class<T> getClassGenricType(final Class clazz)
|
| | | {
|
| | | return getClassGenricType(clazz, 0);
|
| | | }
|
| | |
|
| | | /**
|
| | | * éè¿åå°, è·å¾Classå®ä¹ä¸å£°æçç¶ç±»çæ³ååæ°çç±»å.
|
| | | * 妿 æ³æ¾å°, è¿åObject.class.
|
| | | */
|
| | | public static Class getClassGenricType(final Class clazz, final int index)
|
| | | {
|
| | | Type genType = clazz.getGenericSuperclass();
|
| | |
|
| | | if (!(genType instanceof ParameterizedType))
|
| | | {
|
| | | logger.debug(clazz.getSimpleName() + "'s superclass not ParameterizedType");
|
| | | return Object.class;
|
| | | }
|
| | |
|
| | | Type[] params = ((ParameterizedType) genType).getActualTypeArguments();
|
| | |
|
| | | if (index >= params.length || index < 0)
|
| | | {
|
| | | logger.debug("Index: " + index + ", Size of " + clazz.getSimpleName() + "'s Parameterized Type: "
|
| | | + params.length);
|
| | | return Object.class;
|
| | | }
|
| | | if (!(params[index] instanceof Class))
|
| | | {
|
| | | logger.debug(clazz.getSimpleName() + " not set the actual class on superclass generic parameter");
|
| | | return Object.class;
|
| | | }
|
| | |
|
| | | return (Class) params[index];
|
| | | }
|
| | |
|
| | | public static Class<?> getUserClass(Object instance)
|
| | | {
|
| | | if (instance == null)
|
| | | {
|
| | | throw new RuntimeException("Instance must not be null");
|
| | | }
|
| | | Class clazz = instance.getClass();
|
| | | if (clazz != null && clazz.getName().contains(CGLIB_CLASS_SEPARATOR))
|
| | | {
|
| | | Class<?> superClass = clazz.getSuperclass();
|
| | | if (superClass != null && !Object.class.equals(superClass))
|
| | | {
|
| | | return superClass;
|
| | | }
|
| | | }
|
| | | return clazz;
|
| | |
|
| | | }
|
| | |
|
| | | /**
|
| | | * å°åå°æ¶çchecked exception转æ¢ä¸ºunchecked exception.
|
| | | */
|
| | | public static RuntimeException convertReflectionExceptionToUnchecked(String msg, Exception e)
|
| | | {
|
| | | if (e instanceof IllegalAccessException || e instanceof IllegalArgumentException
|
| | | || e instanceof NoSuchMethodException)
|
| | | {
|
| | | return new IllegalArgumentException(msg, e);
|
| | | }
|
| | | else if (e instanceof InvocationTargetException)
|
| | | {
|
| | | return new RuntimeException(msg, ((InvocationTargetException) e).getTargetException());
|
| | | }
|
| | | return new RuntimeException(msg, e);
|
| | | }
|
| | | }
|