valueOf and Constructor-based conversion of source objects to targetTypes

This commit is contained in:
Keith Donald 2009-11-12 05:25:03 +00:00
parent b5cda8db7b
commit b163f123ce
4 changed files with 118 additions and 3 deletions

View File

@ -36,7 +36,6 @@ public class DefaultConversionService extends GenericConversionService {
addConverter(String.class, Character.class, new StringToCharacterConverter());
addConverter(String.class, Locale.class, new StringToLocaleConverter());
addConverter(Number.class, Character.class, new NumberToCharacterConverter());
addConverter(Object.class, String.class, new ObjectToStringConverter());
addConverterFactory(String.class, Number.class, new StringToNumberConverterFactory());
addConverterFactory(String.class, Enum.class, new StringToEnumConverterFactory());
addConverterFactory(Number.class, Number.class, new NumberToNumberConverterFactory());

View File

@ -80,7 +80,9 @@ public class GenericConversionService implements ConversionService, ConverterReg
addGenericConverter(Map.class, Object.class, new MapToObjectConverter(this));
addGenericConverter(Object.class, Object[].class, new ObjectToArrayConverter(this));
addGenericConverter(Object.class, Collection.class, new ObjectToCollectionConverter(this));
addGenericConverter(Object.class, Map.class, new ObjectToMapConverter(this));
addGenericConverter(Object.class, Map.class, new ObjectToMapConverter(this));
addConverter(Object.class, String.class, new ObjectToStringConverter());
addGenericConverter(Object.class, Object.class, new ObjectToObjectGenericConverter());
}
/**

View File

@ -0,0 +1,70 @@
/*
* Copyright 2002-2009 the original author or authors.
*
* Licensed 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.springframework.core.convert.support;
import java.lang.reflect.Constructor;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import org.springframework.core.convert.ConversionFailedException;
import org.springframework.core.convert.TypeDescriptor;
import org.springframework.util.ClassUtils;
import org.springframework.util.ReflectionUtils;
/**
* Generic Converter that attempts to convert a source Object to a targetType by delegating to methods on the targetType.
* Calls the static valueOf(sourceType) method on the targetType to perform the conversion, if such a method exists.
* Else calls the targetType's Constructor that accepts a single sourceType argument, if such a Constructor exists.
* Else throws a ConversionFailedException.
*
* @author Keith Donald
* @since 3.0
*/
final class ObjectToObjectGenericConverter implements GenericConverter {
public Object convert(Object source, TypeDescriptor sourceType, TypeDescriptor targetType) {
if (sourceType.isAssignableTo(targetType)) {
return source;
}
Class<?> sourceClass = sourceType.getObjectType();
Class<?> targetClass = targetType.getObjectType();
Object target;
Method method = ClassUtils.getStaticMethod(targetClass, "valueOf", sourceClass);
if (method != null) {
target = ReflectionUtils.invokeMethod(method, null, source);
} else {
Constructor<?> constructor = ClassUtils.getConstructorIfAvailable(targetClass, sourceClass);
if (constructor != null) {
try {
target = constructor.newInstance(source);
} catch (IllegalArgumentException e) {
throw new ConversionFailedException(sourceType, targetType, source, e);
} catch (InstantiationException e) {
throw new ConversionFailedException(sourceType, targetType, source, e);
} catch (IllegalAccessException e) {
throw new ConversionFailedException(sourceType, targetType, source, e);
} catch (InvocationTargetException e) {
throw new ConversionFailedException(sourceType, targetType, source, e);
}
} else {
throw new IllegalStateException("No static valueOf(" + sourceClass.getName()
+ ") method or Constructor(" + sourceClass.getName() + ") exists on " + targetClass.getName());
}
}
return target;
}
}

View File

@ -67,6 +67,7 @@ public class GenericConversionServiceTests {
@Test
public void converterNotFound() {
try {
conversionService.removeConvertible(Object.class, Object.class);
conversionService.convert("3", Integer.class);
fail("Should have thrown an exception");
} catch (ConverterNotFoundException e) {
@ -125,6 +126,7 @@ public class GenericConversionServiceTests {
@Test
public void convertObjectToPrimitive() {
conversionService.removeConvertible(Object.class, Object.class);
assertFalse(conversionService.canConvert(String.class, boolean.class));
conversionService.addConverter(new StringToBooleanConverter());
assertTrue(conversionService.canConvert(String.class, boolean.class));
@ -660,6 +662,46 @@ public class GenericConversionServiceTests {
assertEquals(new Long(1), result.get(1L));
}
@Test
public void convertObjectToObjectValueOFMethod() {
assertEquals(new Integer(3), conversionService.convert("3", Integer.class));
}
@Test
public void convertObjectToObjectConstructor() {
assertEquals(new SSN("123456789"), conversionService.convert("123456789", SSN.class));
assertEquals("123456789", conversionService.convert(new SSN("123456789"), String.class));
}
@Test(expected=ConversionFailedException.class)
public void convertObjectToObjectNoValueOFMethodOrConstructor() {
conversionService.convert(new Long(3), Integer.class);
}
private static class SSN {
private String value;
public SSN(String value) {
this.value = value;
}
public boolean equals(Object o) {
if (!(o instanceof SSN)) {
return false;
}
SSN ssn = (SSN) o;
return this.value.equals(ssn.value);
}
public int hashCode() {
return value.hashCode();
}
public String toString() {
return value;
}
}
@Test
public void genericConverterDelegatingBackToConversionServiceConverterNotFound() {
conversionService.addGenericConverter(Object.class, Object[].class, new ObjectToArrayConverter(
@ -675,6 +717,8 @@ public class GenericConversionServiceTests {
public void parent() {
GenericConversionService parent = new GenericConversionService();
conversionService.setParent(parent);
conversionService.removeConvertible(Object.class, Object.class);
parent.removeConvertible(Object.class, Object.class);
assertFalse(conversionService.canConvert(String.class, Integer.class));
try {
conversionService.convert("3", Integer.class);
@ -682,7 +726,7 @@ public class GenericConversionServiceTests {
}
}
public static enum FooEnum {
BAR, BAZ
}