org.apache.camel.util.ReflectionHelper Maven / Gradle / Ivy
/*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to You under the Apache License, Version 2.0
* (the "License"); you may not use this file except in compliance with
* the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.apache.camel.util;
import java.lang.reflect.Field;
import java.lang.reflect.Method;
import java.lang.reflect.Modifier;
import java.util.Arrays;
/**
* Helper for working with reflection on classes.
*
* This code is based on org.apache.camel.spring.util.ReflectionUtils class.
*/
public final class ReflectionHelper {
private ReflectionHelper() {
// utility class
}
/**
* Callback interface invoked on each field in the hierarchy.
*/
@FunctionalInterface
public interface FieldCallback {
/**
* Perform an operation using the given field.
*
* @param field the field to operate on
*/
void doWith(Field field) throws IllegalArgumentException, IllegalAccessException;
}
/**
* Action to take on each method.
*/
@FunctionalInterface
public interface MethodCallback {
/**
* Perform an operation using the given method.
*
* @param method the method to operate on
*/
void doWith(Method method) throws IllegalArgumentException, IllegalAccessException;
}
/**
* Action to take on each class.
*/
@FunctionalInterface
public interface ClassCallback {
/**
* Perform an operation using the given class.
*
* @param clazz the class to operate on
*/
void doWith(Class> clazz) throws IllegalArgumentException, IllegalAccessException;
}
/**
* Perform the given callback operation on the nested (inner) classes.
*
* @param clazz class to start looking at
* @param cc the callback to invoke for each inner class (excluding the class itself)
*/
public static void doWithClasses(Class> clazz, ClassCallback cc) throws IllegalArgumentException {
// and then nested classes
Class>[] classes = clazz.getDeclaredClasses();
for (Class> aClazz : classes) {
try {
cc.doWith(aClazz);
} catch (IllegalAccessException ex) {
throw new IllegalStateException("Shouldn't be illegal to access class '" + aClazz.getName() + "': " + ex);
}
}
}
/**
* Invoke the given callback on all fields in the target class, going up the class hierarchy to get all declared
* fields.
*
* @param clazz the target class to analyze
* @param fc the callback to invoke for each field
*/
public static void doWithFields(Class> clazz, FieldCallback fc) throws IllegalArgumentException {
// Keep backing up the inheritance hierarchy.
Class> targetClass = clazz;
do {
Field[] fields = targetClass.getDeclaredFields();
for (Field field : fields) {
try {
fc.doWith(field);
} catch (IllegalAccessException ex) {
throw new IllegalStateException("Shouldn't be illegal to access field '" + field.getName() + "': " + ex);
}
}
targetClass = targetClass.getSuperclass();
} while (targetClass != null && targetClass != Object.class);
}
/**
* Perform the given callback operation on all matching methods of the given class and superclasses (or given
* interface and super-interfaces).
*
* Important: This method does not take the {@link java.lang.reflect.Method#isBridge() bridge methods} into
* account.
*
* @param clazz class to start looking at
* @param mc the callback to invoke for each method
*/
public static void doWithMethods(Class> clazz, MethodCallback mc) throws IllegalArgumentException {
// Keep backing up the inheritance hierarchy.
Method[] methods = clazz.getDeclaredMethods();
for (Method method : methods) {
if (method.isBridge()) {
// skip the bridge methods which in Java 8 leads to problems with inheritance
// see https://bugs.openjdk.java.net/browse/JDK-6695379
continue;
}
try {
mc.doWith(method);
} catch (IllegalAccessException ex) {
throw new IllegalStateException("Shouldn't be illegal to access method '" + method.getName() + "': " + ex);
}
}
if (clazz.getSuperclass() != null) {
doWithMethods(clazz.getSuperclass(), mc);
} else if (clazz.isInterface()) {
for (Class> superIfc : clazz.getInterfaces()) {
doWithMethods(superIfc, mc);
}
}
}
/**
* Attempt to find a {@link Method} on the supplied class with the supplied name and parameter types. Searches all
* superclasses up to {@code Object}.
*
* Returns {@code null} if no {@link Method} can be found.
*
* @param clazz the class to introspect
* @param name the name of the method
* @param paramTypes the parameter types of the method (may be {@code null} to indicate any signature)
* @return the Method object, or {@code null} if none found
*/
public static Method findMethod(Class> clazz, String name, Class>... paramTypes) {
ObjectHelper.notNull(clazz, "Class must not be null");
ObjectHelper.notNull(name, "Method name must not be null");
Class> searchType = clazz;
while (searchType != null) {
Method[] methods = searchType.isInterface() ? searchType.getMethods() : searchType.getDeclaredMethods();
for (Method method : methods) {
if (name.equals(method.getName())
&& (paramTypes == null || Arrays.equals(paramTypes, method.getParameterTypes()))) {
return method;
}
}
searchType = searchType.getSuperclass();
}
return null;
}
public static void setField(Field f, Object instance, Object value) {
try {
if (!Modifier.isPublic(f.getModifiers()) && !f.canAccess(instance)) {
f.setAccessible(true);
}
// must use fine-grained for the correct type when setting a field value via reflection
Class> type = f.getType();
if (boolean.class == type) {
boolean val;
if (value instanceof Boolean) {
val = (boolean) value;
} else {
val = Boolean.parseBoolean(value.toString());
}
f.setBoolean(instance, val);
} else if (byte.class == type) {
byte val;
if (value instanceof Byte) {
val = (byte) value;
} else {
val = Byte.parseByte(value.toString());
}
f.setByte(instance, val);
} else if (int.class == type) {
int val;
if (value instanceof Integer) {
val = (int) value;
} else {
val = Integer.parseInt(value.toString());
}
f.setInt(instance, val);
} else if (long.class == type) {
long val;
if (value instanceof Long) {
val = (long) value;
} else {
val = Long.parseLong(value.toString());
}
f.setLong(instance, val);
} else if (float.class == type) {
float val;
if (value instanceof Float) {
val = (float) value;
} else {
val = Float.parseFloat(value.toString());
}
f.setFloat(instance, val);
} else if (double.class == type) {
double val;
if (value instanceof Double) {
val = (double) value;
} else {
val = Double.parseDouble(value.toString());
}
f.setDouble(instance, val);
} else {
f.set(instance, value);
}
} catch (Exception ex) {
throw new UnsupportedOperationException("Cannot inject value of class: " + value.getClass() + " into: " + f);
}
}
public static Object getField(Field f, Object instance) {
try {
if (!Modifier.isPublic(f.getModifiers()) && !f.canAccess(instance)) {
f.setAccessible(true);
}
return f.get(instance);
} catch (Exception ex) {
// ignore
}
return null;
}
}