gson-comments/gson/src/main/java/com/google/gson/JsonArrayDeserializationVis...

135 lines
4.8 KiB
Java

/*
* 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.Array;
import java.lang.reflect.Field;
import java.lang.reflect.Type;
import java.util.Collection;
/**
* A visitor that populates fields of an object with data from its equivalent
* JSON representation
*
* @author Inderjeet Singh
* @author Joel Leitch
*/
final class JsonArrayDeserializationVisitor<T> extends JsonDeserializationVisitor<T> {
private final Class<?> componentType;
JsonArrayDeserializationVisitor(JsonArray jsonArray, Type arrayType,
ObjectNavigatorFactory factory, ObjectConstructor objectConstructor,
TypeAdapter typeAdapter, ParameterizedTypeHandlerMap<JsonDeserializer<?>> deserializers,
JsonDeserializationContext context) {
super(jsonArray, arrayType, factory, objectConstructor, typeAdapter, deserializers, context);
this.componentType = TypeUtils.toRawClass(arrayType);
}
@Override
@SuppressWarnings("unchecked")
protected T constructTarget() {
TypeInfo typeInfo = new TypeInfo(targetType);
JsonArray jsonArray = json.getAsJsonArray();
if (typeInfo.isPrimitiveOrStringAndNotAnArray()) {
if (jsonArray.size() != 1) {
throw new IllegalArgumentException(
"Primitives should be an array of length 1, but was: " + jsonArray);
}
return (T) objectConstructor.construct(typeInfo.getWrappedClass());
} else if (typeInfo.isArray()) {
TypeInfoArray arrayTypeInfo = TypeInfoFactory.getTypeInfoForArray(targetType);
// We know that we are getting back an array of the required type, so
// this typecasting is safe.
return (T) objectConstructor.constructArray(arrayTypeInfo.getSecondLevelType(),
jsonArray.size());
} else { // is a collection
return (T) objectConstructor.construct(typeInfo.getRawClass());
}
}
public void visitArray(Object array, Type arrayType) {
JsonArray jsonArray = json.getAsJsonArray();
TypeInfoArray arrayTypeInfo = TypeInfoFactory.getTypeInfoForArray(arrayType);
for (int i = 0; i < jsonArray.size(); i++) {
JsonElement jsonChild = jsonArray.get(i);
Object child;
if (jsonChild == null) {
child = null;
} else if (jsonChild instanceof JsonObject) {
child = visitChildAsObject(arrayTypeInfo.getComponentRawType(), jsonChild);
} else if (jsonChild instanceof JsonArray) {
child = visitChildAsArray(arrayTypeInfo.getSecondLevelType(), jsonChild.getAsJsonArray());
} else if (jsonChild instanceof JsonPrimitive) {
child = visitChildAsPrimitive(arrayTypeInfo.getComponentRawType(),
jsonChild.getAsJsonPrimitive());
} else {
throw new IllegalStateException();
}
Array.set(array, i, child);
}
}
@SuppressWarnings("unchecked")
public void visitCollection(Collection collection, Type collectionType) {
Type childType = TypeUtils.getActualTypeForFirstTypeVariable(collectionType);
for (JsonElement jsonChild : json.getAsJsonArray()) {
if (childType == Object.class) {
throw new JsonParseException(collection +
" must not be a raw collection. Try making it genericized instead.");
}
Object child = visitChild(childType, jsonChild);
collection.add(child);
}
}
@SuppressWarnings("unchecked")
public void visitPrimitiveValue(Object obj) {
target = (T) typeAdapter.adaptType(json.getAsJsonArray().get(0).getAsObject(), componentType);
}
// We should not implement any other method from Visitor interface since
// all other methods should be invoked on JsonObjectDeserializationVisitor
// instead.
public void endVisitingObject(Object node) {
throw new UnsupportedOperationException();
}
public void startVisitingObject(Object node) {
throw new UnsupportedOperationException();
}
public void visitArrayField(Field f, Type typeOfF, Object obj) {
throw new UnsupportedOperationException();
}
public void visitCollectionField(Field f, Type typeOfF, Object obj) {
throw new UnsupportedOperationException();
}
public void visitObjectField(Field f, Type typeOfF, Object obj) {
throw new UnsupportedOperationException();
}
public void visitPrimitiveField(Field f, Type typeOfF, Object obj) {
throw new UnsupportedOperationException();
}
}