/* * Copyright (c) 2006-2012 Rogério Liesenfeld * This file is subject to the terms of the MIT license (see LICENSE.txt). */ package mockit.internal.util; import java.lang.reflect.*; import java.util.*; import mockit.external.asm4.Type; import static java.util.Collections.*; /** * Provides default values for each type, typically used for returning default values according to * method return types. */ @SuppressWarnings("ZeroLengthArrayAllocation") public final class DefaultValues { private DefaultValues() {} private static final Integer ZERO_INT = 0; private static final Long ZERO_LONG = 0L; private static final Float ZERO_FLOAT = 0.0F; private static final Double ZERO_DOUBLE = 0.0; private static final Byte ZERO_BYTE = 0; private static final Short ZERO_SHORT = 0; private static final Character ZERO_CHAR = '\0'; private static final Map<String, Object> TYPE_DESC_TO_VALUE_MAP = new HashMap<String, Object>(); private static final Map<String, Object> ELEM_TYPE_TO_ONE_D_ARRAY = new HashMap<String, Object>(); static { TYPE_DESC_TO_VALUE_MAP.put("Z", Boolean.FALSE); TYPE_DESC_TO_VALUE_MAP.put("C", ZERO_CHAR); TYPE_DESC_TO_VALUE_MAP.put("B", ZERO_BYTE); TYPE_DESC_TO_VALUE_MAP.put("S", ZERO_SHORT); TYPE_DESC_TO_VALUE_MAP.put("I", ZERO_INT); TYPE_DESC_TO_VALUE_MAP.put("F", ZERO_FLOAT); TYPE_DESC_TO_VALUE_MAP.put("J", ZERO_LONG); TYPE_DESC_TO_VALUE_MAP.put("D", ZERO_DOUBLE); TYPE_DESC_TO_VALUE_MAP.put("Ljava/util/Enumeration;", enumeration(emptyList())); TYPE_DESC_TO_VALUE_MAP.put("Ljava/util/Collection;", emptyList()); TYPE_DESC_TO_VALUE_MAP.put("Ljava/util/List;", emptyList()); TYPE_DESC_TO_VALUE_MAP.put("Ljava/util/Set;", emptySet()); TYPE_DESC_TO_VALUE_MAP.put("Ljava/util/SortedSet;", unmodifiableSortedSet(new TreeSet<Object>())); TYPE_DESC_TO_VALUE_MAP.put("Ljava/util/Map;", emptyMap()); TYPE_DESC_TO_VALUE_MAP.put("Ljava/util/SortedMap;", unmodifiableSortedMap(new TreeMap<Object, Object>())); ELEM_TYPE_TO_ONE_D_ARRAY.put("[Z", new boolean[0]); ELEM_TYPE_TO_ONE_D_ARRAY.put("[C", new char[0]); ELEM_TYPE_TO_ONE_D_ARRAY.put("[B", new byte[0]); ELEM_TYPE_TO_ONE_D_ARRAY.put("[S", new short[0]); ELEM_TYPE_TO_ONE_D_ARRAY.put("[I", new int[0]); ELEM_TYPE_TO_ONE_D_ARRAY.put("[F", new float[0]); ELEM_TYPE_TO_ONE_D_ARRAY.put("[J", new long[0]); ELEM_TYPE_TO_ONE_D_ARRAY.put("[D", new double[0]); ELEM_TYPE_TO_ONE_D_ARRAY.put("[Ljava/lang/Object;", new Object[0]); ELEM_TYPE_TO_ONE_D_ARRAY.put("[Ljava/lang/String;", new String[0]); } public static Object computeForReturnType(String methodNameAndDesc) { String typeDesc = getReturnTypeDesc(methodNameAndDesc); return computeForType(typeDesc); } public static String getReturnTypeDesc(String methodNameAndDesc) { if (methodNameAndDesc == null) return null; int rightParen = methodNameAndDesc.indexOf(')') + 1; return methodNameAndDesc.substring(rightParen); } public static Object computeForType(String typeDesc) { char typeDescChar = typeDesc.charAt(0); if (typeDescChar == 'V') { return null; } Object defaultValue = TYPE_DESC_TO_VALUE_MAP.get(typeDesc); if (defaultValue != null) { return defaultValue; } if (typeDescChar == 'L') { return null; } // It's an array. Object emptyArray = ELEM_TYPE_TO_ONE_D_ARRAY.get(typeDesc); if (emptyArray == null) { emptyArray = newEmptyArray(typeDesc); } return emptyArray; } private static Object newEmptyArray(String typeDesc) { Type type = Type.getType(typeDesc); Class<?> elementType = TypeDescriptor.getClassForType(type.getElementType()); return Array.newInstance(elementType, new int[type.getDimensions()]); } public static Object computeForType(Class<?> type) { if (type.isArray()) { return Array.newInstance(type.getComponentType(), 0); } else if (type != void.class && type.isPrimitive()) { return defaultValueForPrimitiveType(type); } return null; } public static Object defaultValueForPrimitiveType(Class<?> type) { if (type == int.class) { return ZERO_INT; } else if (type == boolean.class) { return Boolean.FALSE; } else if (type == long.class) { return ZERO_LONG; } else if (type == double.class) { return ZERO_DOUBLE; } else if (type == float.class) { return ZERO_FLOAT; } else if (type == char.class) { return ZERO_CHAR; } else if (type == byte.class) { return ZERO_BYTE; } else { return ZERO_SHORT; } } @SuppressWarnings("unchecked") public static <T> T computeForWrapperType(java.lang.reflect.Type type) { if (type == Integer.class) { return (T) ZERO_INT; } else if (type == Boolean.class) { return (T) Boolean.FALSE; } else if (type == Long.class) { return (T) ZERO_LONG; } else if (type == Double.class) { return (T) ZERO_DOUBLE; } else if (type == Float.class) { return (T) ZERO_FLOAT; } else if (type == Character.class) { return (T) ZERO_CHAR; } else if (type == Byte.class) { return (T) ZERO_BYTE; } else if (type == Short.class) { return (T) ZERO_SHORT; } return null; } }