19

可以使用 Jackson? 反序列化为具有私有字段和自定义参数构造函数的类,而无需使用注释和修改类

我知道在杰克逊中使用这种组合是可能的:1)Java 8,2)使用“-parameters”选项编译,3)参数名称与 JSON 匹配。但是在没有所有这些限制的情况下,默认情况下在 GSON 中也是可能的。

例如:

public class Person {
    private final String firstName;
    private final String lastName;
    private final int age;

    public Person(String firstName, String lastName, int age) {
        this.firstName = firstName;
        this.lastName = lastName;
        this.age = age;
    }

    public static void main(String[] args) throws IOException {
        String json = "{firstName: \"Foo\", lastName: \"Bar\", age: 30}";
        
        System.out.println("GSON: " + deserializeGson(json)); // works fine
        System.out.println("Jackson: " + deserializeJackson(json)); // error
    }

    public static Person deserializeJackson(String json) throws IOException {
        ObjectMapper mapper = new ObjectMapper();
        mapper.enable(JsonParser.Feature.ALLOW_UNQUOTED_FIELD_NAMES);
        mapper.setVisibility(PropertyAccessor.FIELD, JsonAutoDetect.Visibility.ANY);
        return mapper.readValue(json, Person.class);
    }

    public static Person deserializeGson(String json) {
        Gson gson = new GsonBuilder().create();
        return gson.fromJson(json, Person.class);
    }
}

这适用于 GSON,但杰克逊抛出:

Exception in thread "main" com.fasterxml.jackson.databind.exc.InvalidDefinitionException: Cannot construct instance of `jacksonParametersTest.Person` (no Creators, like default construct, exist): cannot deserialize from Object value (no delegate- or property-based Creator)
 at [Source: (String)"{firstName: "Foo", lastName: "Bar", age: 30}"; line: 1, column: 2]
    at com.fasterxml.jackson.databind.exc.InvalidDefinitionException.from(InvalidDefinitionException.java:67)

这在 GSON 中是可能的,所以我希望 Jackson 中必须有某种方法,而无需修改 Person 类、没有 Java 8,也没有显式的自定义反序列化器。有人知道解决方案吗?

  • 更新,附加信息

Gson 似乎跳过了参数构造函数,因此它必须在幕后使用反射创建一个无参数构造函数。

此外,即使没有“-parameters”编译器标志,也有一个Kotlin Jackson 模块能够为 Kotlin 数据类执行此操作。所以奇怪的是,Java Jackson 似乎不存在这样的解决方案。

这是 Kotlin Jackson 中可用的(漂亮而干净的)解决方案(IMO 也应该通过自定义模块在 Java Jackson 中可用):

val mapper = ObjectMapper()
    .enable(JsonParser.Feature.ALLOW_UNQUOTED_FIELD_NAMES)
    .registerModule(KotlinModule())     
        
val person: Person = mapper.readValue(json, Person::class.java)
4

4 回答 4

13

带有混合注释的解决方案

您可以使用混合注释。当修改类不是一个选项时,这是一个很好的选择。您可以将其视为一种在运行时添加更多注释的面向方面的方式,以增加静态定义的注释。

假设您的Person类定义如下:

public class Person {

    private final String firstName;
    private final String lastName;
    private final int age;

    public Person(String firstName, String lastName, int age) {
        this.firstName = firstName;
        this.lastName = lastName;
        this.age = age;
    }

    // Getters omitted
}

首先定义一个混入注解抽象类:

public abstract class PersonMixIn {

    PersonMixIn(@JsonProperty("firstName") String firstName,
                @JsonProperty("lastName") String lastName,
                @JsonProperty("age") int age) {
    }
}

然后配置ObjectMapper为使用定义的类作为 POJO 的混合:

ObjectMapper mapper = new ObjectMapper();
mapper.enable(JsonParser.Feature.ALLOW_UNQUOTED_FIELD_NAMES);
mapper.addMixIn(Person.class, PersonMixIn.class);

并反序列化 JSON:

String json = "{firstName: \"Foo\", lastName: \"Bar\", age: 30}";
Person person = mapper.readValue(json, Person.class);
于 2017-11-30T22:58:44.643 回答
1

Jackson 提供了模块jackson-modules-java8来解决您的问题。

您必须ObjectMapper按以下方式构建:

ObjectMapper mapper = new ObjectMapper()
        .enable(JsonParser.Feature.ALLOW_UNQUOTED_FIELD_NAMES)
        .registerModule(new ParameterNamesModule(JsonCreator.Mode.PROPERTIES));

您必须添加-parameters为编译器参数。

Maven的示例:

 <plugin>
       <groupId>org.apache.maven.plugins</groupId>
       <artifactId>maven-compiler-plugin</artifactId>
       <version>3.5.1</version>
       <configuration>
           <!--somecode-->

           <compilerArgument>-parameters</compilerArgument>

       </configuration>
 </plugin>

对于毕业:

compileJava {
  options.compilerArgs << "-parameters"
}
于 2017-11-30T19:15:48.323 回答
1

由于没有默认构造函数,jackson 或 gson 希望自己创建实例。你应该告诉 API 如何通过提供自定义反序列化来创建这样的实例。

这是一个片段代码

public class PersonDeserializer extends StdDeserializer<Person> { 
    public PersonDeserializer() {
        super(Person.class);
    } 

    @Override
    public Person deserialize(JsonParser jp, DeserializationContext ctxt) 
            throws IOException, JsonProcessingException {
        try {
            final JsonNode node = jp.getCodec().readTree(jp);
            final ObjectMapper mapper = new ObjectMapper();
            final Person person = (Person) mapper.readValue(node.toString(),
                    Person.class);
            return person;
        } catch (final Exception e) {
            throw new IOException(e);
        }
    }
}

然后注册简单的模块来处理你的类型

final ObjectMapper mapper = jacksonBuilder().build();
SimpleModule module = new SimpleModule();
module.addDeserializer(Person.class, new PersonDeserializer());
于 2017-11-30T13:42:35.143 回答
-1

仅当您可以更改类实现时,以下解决方案才有效

一个简单的解决方案就是创建一个默认构造函数

Person()Person课堂上

public class Person {
    private final String firstName;
    private final String lastName;
    private final int age;

    public Person() {
    }

    public Person(String firstName, String lastName, int age) {
        this.firstName = firstName;
        this.lastName = lastName;
        this.age = age;
    }
    ...
}
于 2019-09-05T13:06:51.933 回答