/* * Copyright (C) 2008 Google Inc. * * 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 com.google.gson; import java.lang.reflect.AccessibleObject; import java.lang.reflect.Array; import java.lang.reflect.Constructor; import java.lang.reflect.InvocationTargetException; import java.lang.reflect.Type; import java.util.logging.Level; import java.util.logging.Logger; /** * This class contains a mapping of all the application specific * {@link InstanceCreator} instances. Registering an {@link InstanceCreator} * with this class will override the default object creation that is defined * by the ObjectConstructor that this class is wrapping. Using this class * with the JSON framework provides the application with "pluggable" modules * to customize framework to suit the application's needs. * * @author Joel Leitch */ final class MappedObjectConstructor implements ObjectConstructor { private final Logger log = Logger.getLogger(getClass().getName()); private final ParameterizedTypeHandlerMap> instanceCreatorMap = new ParameterizedTypeHandlerMap>(); @SuppressWarnings("unchecked") public T construct(Type typeOfT) { if (instanceCreatorMap.hasAnyHandlerFor(typeOfT)) { InstanceCreator creator = (InstanceCreator) instanceCreatorMap.getHandlerFor(typeOfT); return creator.createInstance(typeOfT); } TypeInfo typeInfo = new TypeInfo(typeOfT); if (typeInfo.isEnum()) { InstanceCreator creator = (InstanceCreator) instanceCreatorMap.getHandlerFor(Enum.class); return creator.createInstance(typeOfT); } return (T) constructWithNoArgConstructor(typeOfT); } public Object constructArray(Type type, int length) { return Array.newInstance(TypeUtils.toRawClass(type), length); } private T constructWithNoArgConstructor(Type typeOfT) { try { Constructor constructor = getNoArgsConstructor(typeOfT); if (constructor == null) { throw new RuntimeException(("No-args constructor for " + typeOfT + " does not exist. " + "Register an InstanceCreator with Gson for this type to fix this problem.")); } return constructor.newInstance(); } catch (InstantiationException e) { throw new RuntimeException(("Unable to invoke no-args constructor for " + typeOfT + ". " + "Register an InstanceCreator with Gson for this type may fix this problem."), e); } catch (IllegalAccessException e) { throw new RuntimeException(("Unable to invoke no-args constructor for " + typeOfT + ". " + "Register an InstanceCreator with Gson for this type may fix this problem."), e); } catch (InvocationTargetException e) { throw new RuntimeException(("Unable to invoke no-args constructor for " + typeOfT + ". " + "Register an InstanceCreator with Gson for this type may fix this problem."), e); } } @SuppressWarnings({"unchecked", "cast"}) private Constructor getNoArgsConstructor(Type typeOfT) { TypeInfo typeInfo = new TypeInfo(typeOfT); Class clazz = (Class) typeInfo.getRawClass(); Constructor[] declaredConstructors = (Constructor[]) clazz.getDeclaredConstructors(); AccessibleObject.setAccessible(declaredConstructors, true); for (Constructor constructor : declaredConstructors) { if (constructor.getParameterTypes().length == 0) { return constructor; } } return null; } /** * Use this methods to register an {@link InstanceCreator} for a new type. * * @param the type of class to be mapped with its "creator" * @param typeOfT the instance type that will be created * @param creator the {@link InstanceCreator} instance to register */ void register(Type typeOfT, InstanceCreator creator) { if (instanceCreatorMap.hasSpecificHandlerFor(typeOfT)) { log.log(Level.WARNING, "Overriding the existing InstanceCreator for " + typeOfT); } instanceCreatorMap.register(typeOfT, creator); } @Override public String toString() { return instanceCreatorMap.toString(); } }