Po długim kopanie w wiosenno-MongoDB kod kalkulator, Ok skończyłem i teraz wszystko działa :) tu jest (jeśli istnieje prostsze rozwiązanie i będzie zadowolony zobaczyć, jak dobrze, to jest co zrobiłem):
najpierw zdefiniować:
public interface IntEnumConvertable {
public int getValue();
}
i prostego wyliczenia, który implementuje to:
public enum tester implements IntEnumConvertable{
vali(0),secondvali(1),thirdvali(5);
private final int val;
private tester(int num)
{
val = num;
}
public int getValue(){
return val;
}
}
Ok, teraz będzie teraz trzeba 2 konwerterów, jeden jest prosty, druga jest bardziej złożona. ten prosty (to proste dziecko obsługuje również zwykłą konwersję i zwraca ciąg, gdy rzutowanie nie jest możliwe, to jest świetne, jeśli chcesz, aby enum było przechowywane jako łańcuchy znaków i dla wyliczenia, które są liczbami, które mają być przechowywane jako liczby całkowite):
public class IntegerEnumConverters {
@WritingConverter
public static class EnumToIntegerConverter implements Converter<Enum<?>, Object> {
@Override
public Object convert(Enum<?> source) {
if(source instanceof IntEnumConvertable)
{
return ((IntEnumConvertable)(source)).getValue();
}
else
{
return source.name();
}
}
}
}
tym bardziej skomplikowana, w rzeczywistości jest fabrycznie konwerter:
public class IntegerToEnumConverterFactory implements ConverterFactory<Integer, Enum> {
@Override
public <T extends Enum> Converter<Integer, T> getConverter(Class<T> 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 new IntegerToEnum(enumType);
}
@ReadingConverter
public static class IntegerToEnum<T extends Enum> implements Converter<Integer, Enum> {
private final Class<T> enumType;
public IntegerToEnum(Class<T> enumType) {
this.enumType = enumType;
}
@Override
public Enum convert(Integer source) {
for(T t : enumType.getEnumConstants()) {
if(t instanceof IntEnumConvertable)
{
if(((IntEnumConvertable)t).getValue() == source.intValue()) {
return t;
}
}
}
return null;
}
}
}
i teraz dla części hack, ja PERSONNALY zrobił znaleźć żadnego „programmitacly” drogę do zarejestrowania zakładu przetwarzającego obrębie mongoConverter, więc wykopany w kodzie i trochę rzucony, oto jest (umieść to 2 funkcje niemowlęcia w twojej klasie @Configuration)
@Bean
public CustomConversions customConversions() {
List<Converter<?, ?>> converters = new ArrayList<Converter<?, ?>>();
converters.add(new IntegerEnumConverters.EnumToIntegerConverter());
// this is a dummy registration , actually it's a work-around because
// spring-mongodb doesnt has the option to reg converter factory.
// so we reg the converter that our factory uses.
converters.add(new IntegerToEnumConverterFactory.IntegerToEnum(null));
return new CustomConversions(converters);
}
@Bean
public MappingMongoConverter mappingMongoConverter() throws Exception {
MongoMappingContext mappingContext = new MongoMappingContext();
mappingContext.setApplicationContext(appContext);
DbRefResolver dbRefResolver = new DefaultDbRefResolver(mongoDbFactory());
MappingMongoConverter mongoConverter = new MappingMongoConverter(dbRefResolver, mappingContext);
mongoConverter.setCustomConversions(customConversions());
ConversionService convService = mongoConverter.getConversionService();
((GenericConversionService)convService).addConverterFactory(new IntegerToEnumConverterFactory());
mongoConverter.afterPropertiesSet();
return mongoConverter;
}
Dzięki. Użyłem pomysłu na implementację LowerStringToEnumConverter – Demwis
Dziękuję bardzo! To jest bardzo przewodowe, że dane z wiosny nie dają prostego sposobu na zrobienie tego. – ruX
To było całkiem pomocne. –