Merge pull request #998 from quaff/SPR-14039
* pr/998: Polish contribution Add converters between Enum and Integer
This commit is contained in:
commit
1c3c88f81e
|
@ -0,0 +1,48 @@
|
||||||
|
/*
|
||||||
|
* Copyright 2002-2016 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 org.springframework.core.convert.ConversionService;
|
||||||
|
import org.springframework.core.convert.TypeDescriptor;
|
||||||
|
import org.springframework.core.convert.converter.ConditionalConverter;
|
||||||
|
import org.springframework.util.ClassUtils;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A {@link ConditionalConverter} base implementation for enum-based converters.
|
||||||
|
*
|
||||||
|
* @author Stephane Nicoll
|
||||||
|
* @since 4.3
|
||||||
|
*/
|
||||||
|
abstract class AbstractConditionalEnumConverter implements ConditionalConverter {
|
||||||
|
|
||||||
|
private final ConversionService conversionService;
|
||||||
|
|
||||||
|
protected AbstractConditionalEnumConverter(ConversionService conversionService) {
|
||||||
|
this.conversionService = conversionService;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean matches(TypeDescriptor sourceType, TypeDescriptor targetType) {
|
||||||
|
for (Class<?> interfaceType : ClassUtils.getAllInterfacesForClass(sourceType.getType())) {
|
||||||
|
if (this.conversionService.canConvert(TypeDescriptor.valueOf(interfaceType), targetType)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright 2002-2011 the original author or authors.
|
* Copyright 2002-2016 the original author or authors.
|
||||||
*
|
*
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
* you may not use this file except in compliance with the License.
|
* you may not use this file except in compliance with the License.
|
||||||
|
@ -25,6 +25,7 @@ import org.springframework.core.convert.converter.GenericConverter;
|
||||||
* Internal utilities for the conversion package.
|
* Internal utilities for the conversion package.
|
||||||
*
|
*
|
||||||
* @author Keith Donald
|
* @author Keith Donald
|
||||||
|
* @author Stephane Nicoll
|
||||||
* @since 3.0
|
* @since 3.0
|
||||||
*/
|
*/
|
||||||
abstract class ConversionUtils {
|
abstract class ConversionUtils {
|
||||||
|
@ -65,4 +66,16 @@ abstract class ConversionUtils {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public static Class<?> getEnumType(Class<?> targetType) {
|
||||||
|
Class<?> enumType = targetType;
|
||||||
|
while (enumType != null && !enumType.isEnum()) {
|
||||||
|
enumType = enumType.getSuperclass();
|
||||||
|
}
|
||||||
|
if (enumType == null) {
|
||||||
|
throw new IllegalArgumentException(
|
||||||
|
"The target type " + targetType.getName() + " does not refer to an enum");
|
||||||
|
}
|
||||||
|
return enumType;
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -143,6 +143,9 @@ public class DefaultConversionService extends GenericConversionService {
|
||||||
converterRegistry.addConverterFactory(new StringToEnumConverterFactory());
|
converterRegistry.addConverterFactory(new StringToEnumConverterFactory());
|
||||||
converterRegistry.addConverter(new EnumToStringConverter((ConversionService) converterRegistry));
|
converterRegistry.addConverter(new EnumToStringConverter((ConversionService) converterRegistry));
|
||||||
|
|
||||||
|
converterRegistry.addConverterFactory(new IntegerToEnumConverterFactory());
|
||||||
|
converterRegistry.addConverter(new EnumToIntegerConverter((ConversionService) converterRegistry));
|
||||||
|
|
||||||
converterRegistry.addConverter(new StringToLocaleConverter());
|
converterRegistry.addConverter(new StringToLocaleConverter());
|
||||||
converterRegistry.addConverter(Locale.class, String.class, new ObjectToStringConverter());
|
converterRegistry.addConverter(Locale.class, String.class, new ObjectToStringConverter());
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,40 @@
|
||||||
|
/*
|
||||||
|
* Copyright 2002-2016 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 org.springframework.core.convert.ConversionService;
|
||||||
|
import org.springframework.core.convert.converter.Converter;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Calls {@link Enum#ordinal()} to convert a source Enum to a Integer.
|
||||||
|
* This converter will not match enums with interfaces that can be converted.
|
||||||
|
*
|
||||||
|
* @author Yanming Zhou
|
||||||
|
* @since 4.3
|
||||||
|
*/
|
||||||
|
final class EnumToIntegerConverter extends AbstractConditionalEnumConverter implements Converter<Enum<?>, Integer> {
|
||||||
|
|
||||||
|
public EnumToIntegerConverter(ConversionService conversionService) {
|
||||||
|
super(conversionService);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Integer convert(Enum<?> source) {
|
||||||
|
return source.ordinal();
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -17,10 +17,7 @@
|
||||||
package org.springframework.core.convert.support;
|
package org.springframework.core.convert.support;
|
||||||
|
|
||||||
import org.springframework.core.convert.ConversionService;
|
import org.springframework.core.convert.ConversionService;
|
||||||
import org.springframework.core.convert.TypeDescriptor;
|
|
||||||
import org.springframework.core.convert.converter.ConditionalConverter;
|
|
||||||
import org.springframework.core.convert.converter.Converter;
|
import org.springframework.core.convert.converter.Converter;
|
||||||
import org.springframework.util.ClassUtils;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Calls {@link Enum#name()} to convert a source Enum to a String.
|
* Calls {@link Enum#name()} to convert a source Enum to a String.
|
||||||
|
@ -30,24 +27,10 @@ import org.springframework.util.ClassUtils;
|
||||||
* @author Phillip Webb
|
* @author Phillip Webb
|
||||||
* @since 3.0
|
* @since 3.0
|
||||||
*/
|
*/
|
||||||
final class EnumToStringConverter implements Converter<Enum<?>, String>, ConditionalConverter {
|
final class EnumToStringConverter extends AbstractConditionalEnumConverter implements Converter<Enum<?>, String> {
|
||||||
|
|
||||||
private final ConversionService conversionService;
|
|
||||||
|
|
||||||
|
|
||||||
public EnumToStringConverter(ConversionService conversionService) {
|
public EnumToStringConverter(ConversionService conversionService) {
|
||||||
this.conversionService = conversionService;
|
super(conversionService);
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public boolean matches(TypeDescriptor sourceType, TypeDescriptor targetType) {
|
|
||||||
for (Class<?> interfaceType : ClassUtils.getAllInterfacesForClass(sourceType.getType())) {
|
|
||||||
if (this.conversionService.canConvert(TypeDescriptor.valueOf(interfaceType), targetType)) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return true;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
|
|
@ -0,0 +1,52 @@
|
||||||
|
/*
|
||||||
|
* Copyright 2002-2016 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 org.springframework.core.convert.converter.Converter;
|
||||||
|
import org.springframework.core.convert.converter.ConverterFactory;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Converts from a Integer to a {@link java.lang.Enum} by calling {@link Class#getEnumConstants()}.
|
||||||
|
*
|
||||||
|
* @author Yanming Zhou
|
||||||
|
* @author Stephane Nicoll
|
||||||
|
* @since 4.3
|
||||||
|
*/
|
||||||
|
@SuppressWarnings({"unchecked", "rawtypes"})
|
||||||
|
final class IntegerToEnumConverterFactory implements ConverterFactory<Integer, Enum> {
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public <T extends Enum> Converter<Integer, T> getConverter(Class<T> targetType) {
|
||||||
|
return new IntegerToEnum(ConversionUtils.getEnumType(targetType));
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
private class IntegerToEnum<T extends Enum> implements Converter<Integer, T> {
|
||||||
|
|
||||||
|
private final Class<T> enumType;
|
||||||
|
|
||||||
|
public IntegerToEnum(Class<T> enumType) {
|
||||||
|
this.enumType = enumType;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public T convert(Integer source) {
|
||||||
|
return this.enumType.getEnumConstants()[source];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright 2002-2014 the original author or authors.
|
* Copyright 2002-2016 the original author or authors.
|
||||||
*
|
*
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
* you may not use this file except in compliance with the License.
|
* you may not use this file except in compliance with the License.
|
||||||
|
@ -23,6 +23,7 @@ import org.springframework.core.convert.converter.ConverterFactory;
|
||||||
* Converts from a String to a {@link java.lang.Enum} by calling {@link Enum#valueOf(Class, String)}.
|
* Converts from a String to a {@link java.lang.Enum} by calling {@link Enum#valueOf(Class, String)}.
|
||||||
*
|
*
|
||||||
* @author Keith Donald
|
* @author Keith Donald
|
||||||
|
* @author Stephane Nicoll
|
||||||
* @since 3.0
|
* @since 3.0
|
||||||
*/
|
*/
|
||||||
@SuppressWarnings({"unchecked", "rawtypes"})
|
@SuppressWarnings({"unchecked", "rawtypes"})
|
||||||
|
@ -30,15 +31,7 @@ final class StringToEnumConverterFactory implements ConverterFactory<String, Enu
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public <T extends Enum> Converter<String, T> getConverter(Class<T> targetType) {
|
public <T extends Enum> Converter<String, T> getConverter(Class<T> targetType) {
|
||||||
Class<?> enumType = targetType;
|
return new StringToEnum(ConversionUtils.getEnumType(targetType));
|
||||||
while (enumType != null && !enumType.isEnum()) {
|
|
||||||
enumType = enumType.getSuperclass();
|
|
||||||
}
|
|
||||||
if (enumType == null) {
|
|
||||||
throw new IllegalArgumentException(
|
|
||||||
"The target type " + targetType.getName() + " does not refer to an enum");
|
|
||||||
}
|
|
||||||
return new StringToEnum(enumType);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright 2002-2015 the original author or authors.
|
* Copyright 2002-2016 the original author or authors.
|
||||||
*
|
*
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
* you may not use this file except in compliance with the License.
|
* you may not use this file except in compliance with the License.
|
||||||
|
@ -244,6 +244,26 @@ public class DefaultConversionServiceTests {
|
||||||
assertEquals("BAR", conversionService.convert(Foo.BAR, String.class));
|
assertEquals("BAR", conversionService.convert(Foo.BAR, String.class));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testIntegerToEnum() throws Exception {
|
||||||
|
assertEquals(Foo.BAR, conversionService.convert(0, Foo.class));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testIntegerToEnumWithSubclass() throws Exception {
|
||||||
|
assertEquals(SubFoo.BAZ, conversionService.convert(1, SubFoo.BAR.getClass()));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testIntegerToEnumNull() {
|
||||||
|
assertEquals(null, conversionService.convert(null, Foo.class));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testEnumToInteger() {
|
||||||
|
assertEquals(Integer.valueOf(0), conversionService.convert(Foo.BAR, Integer.class));
|
||||||
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void testStringToEnumSet() throws Exception {
|
public void testStringToEnumSet() throws Exception {
|
||||||
assertEquals(EnumSet.of(Foo.BAR), conversionService.convert("BAR", TypeDescriptor.valueOf(String.class),
|
assertEquals(EnumSet.of(Foo.BAR), conversionService.convert("BAR", TypeDescriptor.valueOf(String.class),
|
||||||
|
|
Loading…
Reference in New Issue