Allow MapToMap conversion to work when the target map does not have a
default constructor (as long as a new map copy is not required). Issue: SPR-9284
This commit is contained in:
parent
60afa5ead4
commit
38c4393592
|
@ -1,5 +1,5 @@
|
|||
/*
|
||||
* Copyright 2002-2011 the original author or authors.
|
||||
* Copyright 2002-2012 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.
|
||||
|
@ -16,7 +16,9 @@
|
|||
|
||||
package org.springframework.core.convert.support;
|
||||
|
||||
import java.util.ArrayList;
|
||||
import java.util.Collections;
|
||||
import java.util.List;
|
||||
import java.util.Map;
|
||||
import java.util.Set;
|
||||
|
||||
|
@ -62,18 +64,25 @@ final class MapToMapConverter implements ConditionalGenericConverter {
|
|||
if (!copyRequired && sourceMap.isEmpty()) {
|
||||
return sourceMap;
|
||||
}
|
||||
Map<Object, Object> targetMap = CollectionFactory.createMap(targetType.getType(), sourceMap.size());
|
||||
List<MapEntry> targetEntries = new ArrayList<MapEntry>(sourceMap.size());
|
||||
for (Map.Entry<Object, Object> entry : sourceMap.entrySet()) {
|
||||
Object sourceKey = entry.getKey();
|
||||
Object sourceValue = entry.getValue();
|
||||
Object targetKey = convertKey(sourceKey, sourceType, targetType.getMapKeyTypeDescriptor());
|
||||
Object targetValue = convertValue(sourceValue, sourceType, targetType.getMapValueTypeDescriptor());
|
||||
targetMap.put(targetKey, targetValue);
|
||||
targetEntries.add(new MapEntry(targetKey, targetValue));
|
||||
if (sourceKey != targetKey || sourceValue != targetValue) {
|
||||
copyRequired = true;
|
||||
}
|
||||
}
|
||||
return (copyRequired ? targetMap : sourceMap);
|
||||
if(!copyRequired) {
|
||||
return sourceMap;
|
||||
}
|
||||
Map<Object, Object> targetMap = CollectionFactory.createMap(targetType.getType(), sourceMap.size());
|
||||
for (MapEntry entry : targetEntries) {
|
||||
entry.addToMap(targetMap);
|
||||
}
|
||||
return targetMap;
|
||||
}
|
||||
|
||||
// internal helpers
|
||||
|
@ -102,4 +111,19 @@ final class MapToMapConverter implements ConditionalGenericConverter {
|
|||
return this.conversionService.convert(sourceValue, sourceType.getMapValueTypeDescriptor(sourceValue), targetType);
|
||||
}
|
||||
|
||||
private static class MapEntry {
|
||||
|
||||
private Object key;
|
||||
private Object value;
|
||||
|
||||
public MapEntry(Object key, Object value) {
|
||||
this.key = key;
|
||||
this.value = value;
|
||||
}
|
||||
|
||||
public void addToMap(Map<Object, Object> map) {
|
||||
map.put(key, value);
|
||||
}
|
||||
}
|
||||
|
||||
}
|
||||
|
|
|
@ -1,5 +1,5 @@
|
|||
/*
|
||||
* Copyright 2002-2011 the original author or authors.
|
||||
* Copyright 2002-2012 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.
|
||||
|
@ -17,6 +17,7 @@
|
|||
package org.springframework.core.convert.support;
|
||||
|
||||
import java.util.Arrays;
|
||||
import java.util.Collections;
|
||||
import java.util.HashMap;
|
||||
import java.util.LinkedHashMap;
|
||||
import java.util.List;
|
||||
|
@ -200,4 +201,26 @@ public class MapToMapConverterTests {
|
|||
|
||||
public LinkedHashMap<String, String> emptyMapDifferentTarget;
|
||||
|
||||
@Test
|
||||
public void noDefaultConstructorCopyNotRequired() throws Exception {
|
||||
// SPR-9284
|
||||
NoDefaultConstructorMap<String, Integer> map = new NoDefaultConstructorMap<String,Integer>(
|
||||
Collections.<String, Integer> singletonMap("1", 1));
|
||||
TypeDescriptor sourceType = TypeDescriptor.map(NoDefaultConstructorMap.class,
|
||||
TypeDescriptor.valueOf(String.class), TypeDescriptor.valueOf(Integer.class));
|
||||
TypeDescriptor targetType = TypeDescriptor.map(NoDefaultConstructorMap.class,
|
||||
TypeDescriptor.valueOf(String.class), TypeDescriptor.valueOf(Integer.class));
|
||||
assertTrue(conversionService.canConvert(sourceType, targetType));
|
||||
@SuppressWarnings("unchecked")
|
||||
Map<String, Integer> result = (Map<String, Integer>) conversionService.convert(map, sourceType, targetType);
|
||||
assertEquals(map, result);
|
||||
assertEquals(NoDefaultConstructorMap.class, result.getClass());
|
||||
}
|
||||
|
||||
public static class NoDefaultConstructorMap<K, V> extends HashMap<K, V> {
|
||||
public NoDefaultConstructorMap(Map<? extends K, ? extends V> m) {
|
||||
super(m);
|
||||
}
|
||||
}
|
||||
|
||||
}
|
||||
|
|
Loading…
Reference in New Issue