По умолчанию Gson использует поля в качестве основы для его сериализации. Есть ли способ заставить его использовать аксессоры вместо этого?
Как я могу заставить Gson использовать аксессоры, а не поля?
Ответ 1
Разработчики Gson скажут, что они никогда не чувствовали колебания в запросах на добавление этой функции, и они беспокоились о том, чтобы мутировать api, чтобы добавить поддержка этого.
Одним из способов добавления этой функции является использование TypeAdapter (я извиняюсь за gnarly-код, но это демонстрирует принцип):
import java.io.IOException;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import com.google.common.base.CaseFormat;
import com.google.gson.Gson;
import com.google.gson.TypeAdapter;
import com.google.gson.reflect.TypeToken;
import com.google.gson.stream.JsonReader;
import com.google.gson.stream.JsonWriter;
public class AccessorBasedTypeAdaptor<T> extends TypeAdapter<T> {
private Gson gson;
public AccessorBasedTypeAdaptor(Gson gson) {
this.gson = gson;
}
@SuppressWarnings("unchecked")
@Override
public void write(JsonWriter out, T value) throws IOException {
out.beginObject();
for (Method method : value.getClass().getMethods()) {
boolean nonBooleanAccessor = method.getName().startsWith("get");
boolean booleanAccessor = method.getName().startsWith("is");
if ((nonBooleanAccessor || booleanAccessor) && !method.getName().equals("getClass") && method.getParameterTypes().length == 0) {
try {
String name = method.getName().substring(nonBooleanAccessor ? 3 : 2);
name = CaseFormat.UPPER_CAMEL.to(CaseFormat.LOWER_CAMEL, name);
Object returnValue = method.invoke(value);
if(returnValue != null) {
TypeToken<?> token = TypeToken.get(returnValue.getClass());
TypeAdapter adapter = gson.getAdapter(token);
out.name(name);
adapter.write(out, returnValue);
}
} catch (Exception e) {
throw new ConfigurationException("problem writing json: ", e);
}
}
}
out.endObject();
}
@Override
public T read(JsonReader in) throws IOException {
throw new UnsupportedOperationException("Only supports writes.");
}
}
Вы можете зарегистрировать это как адаптер нормального типа для заданного типа или через TypeAdapterfactory - возможно, для проверки наличия аннотации времени выполнения:
public class TypeFactory implements TypeAdapterFactory {
@SuppressWarnings("unchecked")
public <T> TypeAdapter<T> create(final Gson gson, final TypeToken<T> type) {
Class<? super T> t = type.getRawType();
if(t.isAnnotationPresent(UseAccessor.class)) {
return (TypeAdapter<T>) new AccessorBasedTypeAdaptor(gson);
}
return null;
}
Это может быть указано как обычно при создании экземпляра gson:
new GsonBuilder().registerTypeAdapterFactory(new TypeFactory()).create();
Ответ 2
Примечание. Я EclipseLink JAXB (MOXy) и член группы JAXB (JSR-222).
Если вы не можете заставить Gson делать то, что хотите, ниже вы можете сделать это, используя MOXy native JSON binding. MOXy, как и любая реализация JAXB, будет использовать доступ к свойствам (общедоступным) по умолчанию. Вы можете настроить доступ к полю с помощью @XmlAccessorType(XmlAccessType.FIELD)
. Ниже приведен пример:
Клиент
package forum11385214;
public class Customer {
private String foo;
private Address bar;
public String getName() {
return foo;
}
public void setName(String name) {
this.foo = name;
}
public Address getAddress() {
return bar;
}
public void setAddress(Address address) {
this.bar = address;
}
}
Адрес
package forum11385214;
public class Address {
private String foo;
public String getStreet() {
return foo;
}
public void setStreet(String street) {
this.foo = street;
}
}
jaxb.properties
Чтобы настроить MOXy как поставщика JAXB, вам нужно добавить файл с именем jaxb.properties
в том же пакете, что и ваша модель домена, со следующей записью (см.: <а2 > ).
javax.xml.bind.context.factory=org.eclipse.persistence.jaxb.JAXBContextFactory
Demo
package forum11385214;
import java.util.*;
import javax.xml.bind.*;
import javax.xml.transform.stream.StreamSource;
import org.eclipse.persistence.jaxb.JAXBContextProperties;
public class Demo {
public static void main(String[] args) throws Exception {
Map<String, Object> properties = new HashMap<String, Object>(2);
properties.put(JAXBContextProperties.MEDIA_TYPE, "application/json");
properties.put(JAXBContextProperties.JSON_INCLUDE_ROOT, false);
JAXBContext jc = JAXBContext.newInstance(new Class[] {Customer.class}, properties);
Unmarshaller unmarshaller = jc.createUnmarshaller();
StreamSource json = new StreamSource("src/forum11385214/input.json");
Customer customer = (Customer) unmarshaller.unmarshal(json, Customer.class).getValue();
Marshaller marshaller = jc.createMarshaller();
marshaller.setProperty(Marshaller.JAXB_FORMATTED_OUTPUT, true);
marshaller.marshal(customer, System.out);
}
}
input.json/выход
{
"name" : "Jane Doe",
"address" : {
"street" : "1 Any Street"
}
}
Дополнительная информация