mappedpropertydescriptor.java
来自「这是一个有关common beanutils 的源码」· Java 代码 · 共 508 行 · 第 1/2 页
JAVA
508 行
findMappedPropertyType();
}
/**
* Gets the method that should be used to write one of the property value.
*
* @return The method that should be used to write one of the property value.
* May return null if the property can't be written.
*/
public Method getMappedWriteMethod() {
return mappedWriteMethodRef.get();
}
/**
* Sets the method that should be used to write the property value.
*
* @param mappedSetter The mapped setter method.
* @throws IntrospectionException If an error occurs finding the
* mapped property
*/
public void setMappedWriteMethod(Method mappedSetter)
throws IntrospectionException {
mappedWriteMethodRef = new MappedMethodReference(mappedSetter);
findMappedPropertyType();
}
// ------------------------------------------------------- Private Methods
/**
* Introspect our bean class to identify the corresponding getter
* and setter methods.
*/
private void findMappedPropertyType() throws IntrospectionException {
try {
Method mappedReadMethod = getMappedReadMethod();
Method mappedWriteMethod = getMappedWriteMethod();
Class mappedPropertyType = null;
if (mappedReadMethod != null) {
if (mappedReadMethod.getParameterTypes().length != 1) {
throw new IntrospectionException
("bad mapped read method arg count");
}
mappedPropertyType = mappedReadMethod.getReturnType();
if (mappedPropertyType == Void.TYPE) {
throw new IntrospectionException
("mapped read method " +
mappedReadMethod.getName() + " returns void");
}
}
if (mappedWriteMethod != null) {
Class[] params = mappedWriteMethod.getParameterTypes();
if (params.length != 2) {
throw new IntrospectionException
("bad mapped write method arg count");
}
if (mappedPropertyType != null &&
mappedPropertyType != params[1]) {
throw new IntrospectionException
("type mismatch between mapped read and write methods");
}
mappedPropertyType = params[1];
}
mappedPropertyTypeRef = new SoftReference(mappedPropertyType);
} catch (IntrospectionException ex) {
throw ex;
}
}
/**
* Return a capitalized version of the specified property name.
*
* @param s The property name
*/
private static String capitalizePropertyName(String s) {
if (s.length() == 0) {
return s;
}
char[] chars = s.toCharArray();
chars[0] = Character.toUpperCase(chars[0]);
return new String(chars);
}
/**
* Find a method on a class with a specified number of parameters.
*/
private static Method internalGetMethod(Class initial, String methodName,
int parameterCount) {
// For overridden methods we need to find the most derived version.
// So we start with the given class and walk up the superclass chain.
for (Class clazz = initial; clazz != null; clazz = clazz.getSuperclass()) {
Method[] methods = clazz.getDeclaredMethods();
for (int i = 0; i < methods.length; i++) {
Method method = methods[i];
if (method == null) {
continue;
}
// skip static methods.
int mods = method.getModifiers();
if (!Modifier.isPublic(mods) ||
Modifier.isStatic(mods)) {
continue;
}
if (method.getName().equals(methodName) &&
method.getParameterTypes().length == parameterCount) {
return method;
}
}
}
// Now check any inherited interfaces. This is necessary both when
// the argument class is itself an interface, and when the argument
// class is an abstract class.
Class[] interfaces = initial.getInterfaces();
for (int i = 0; i < interfaces.length; i++) {
Method method = internalGetMethod(interfaces[i], methodName, parameterCount);
if (method != null) {
return method;
}
}
return null;
}
/**
* Find a method on a class with a specified number of parameters.
*/
private static Method getMethod(Class clazz, String methodName, int parameterCount)
throws IntrospectionException {
if (methodName == null) {
return null;
}
Method method = internalGetMethod(clazz, methodName, parameterCount);
if (method != null) {
return method;
}
// No Method found
throw new IntrospectionException("No method \"" + methodName +
"\" with " + parameterCount + " parameter(s)");
}
/**
* Find a method on a class with a specified parameter list.
*/
private static Method getMethod(Class clazz, String methodName, Class[] parameterTypes)
throws IntrospectionException {
if (methodName == null) {
return null;
}
Method method = MethodUtils.getMatchingAccessibleMethod(clazz, methodName, parameterTypes);
if (method != null) {
return method;
}
int parameterCount = (parameterTypes == null) ? 0 : parameterTypes.length;
// No Method found
throw new IntrospectionException("No method \"" + methodName +
"\" with " + parameterCount + " parameter(s) of matching types.");
}
/**
* Holds a {@link Method} in a {@link SoftReference} so that it
* it doesn't prevent any ClassLoader being garbage collected, but
* tries to re-create the method if the method reference has been
* released.
*
* See http://issues.apache.org/jira/browse/BEANUTILS-291
*/
private static class MappedMethodReference {
private String className;
private String methodName;
private Reference methodRef;
private Reference classRef;
private Reference writeParamTypeRef;
MappedMethodReference(Method m) {
if (m != null) {
className = m.getDeclaringClass().getName();
methodName = m.getName();
methodRef = new SoftReference(m);
classRef = new WeakReference(m.getDeclaringClass());
Class[] types = m.getParameterTypes();
if (types.length == 2) {
writeParamTypeRef = new WeakReference(types[1]);
}
}
}
private Method get() {
if (methodRef == null) {
return null;
}
Method m = (Method)methodRef.get();
if (m == null) {
Class clazz = (Class)classRef.get();
if (clazz == null) {
clazz = reLoadClass();
if (clazz != null) {
classRef = new WeakReference(clazz);
}
}
if (clazz == null) {
throw new RuntimeException("Method " + methodName + " for " +
className + " could not be reconstructed - class reference has gone");
}
Class[] paramTypes = null;
if (writeParamTypeRef != null) {
paramTypes = new Class[] {String.class, (Class)writeParamTypeRef.get()};
} else {
paramTypes = STRING_CLASS_PARAMETER;
}
try {
m = clazz.getMethod(methodName, paramTypes);
// Un-comment following line for testing
// System.out.println("Recreated Method " + methodName + " for " + className);
} catch (NoSuchMethodException e) {
throw new RuntimeException("Method " + methodName + " for " +
className + " could not be reconstructed - method not found");
}
methodRef = new SoftReference(m);
}
return m;
}
/**
* Try to re-load the class
*/
private Class reLoadClass() {
ClassLoader classLoader = Thread.currentThread().getContextClassLoader();
// Try the context class loader
if (classLoader != null) {
try {
return classLoader.loadClass(className);
} catch (Throwable t) {
// ignore
}
}
// Try this class's class loader
try {
return classLoader.loadClass(className);
} catch (Throwable t) {
return null;
}
}
}
}
⌨️ 快捷键说明
复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?