我需要将某个JSON字符串转换为Java对象。我正在使用Jackson进行JSON处理。我无法控制输入JSON(我从web服务读取)。这是我的输入JSON:

{"wrapper":[{"id":"13","name":"Fred"}]}

下面是一个简化的用例:

private void tryReading() {
    String jsonStr = "{\"wrapper\"\:[{\"id\":\"13\",\"name\":\"Fred\"}]}";
    ObjectMapper mapper = new ObjectMapper();  
    Wrapper wrapper = null;
    try {
        wrapper = mapper.readValue(jsonStr , Wrapper.class);
    } catch (Exception e) {
        e.printStackTrace();
    }
    System.out.println("wrapper = " + wrapper);
}

我的实体类是:

public Class Student { 
    private String name;
    private String id;
    //getters & setters for name & id here
}

我的Wrapper类基本上是一个容器对象来获取我的学生列表:

public Class Wrapper {
    private List<Student> students;
    //getters & setters here
}

我一直得到这个错误和“包装器”返回null。我不知道少了什么。有人能帮帮我吗?

org.codehaus.jackson.map.exc.UnrecognizedPropertyException: 
    Unrecognized field "wrapper" (Class Wrapper), not marked as ignorable
 at [Source: java.io.StringReader@1198891; line: 1, column: 13] 
    (through reference chain: Wrapper["wrapper"])
 at org.codehaus.jackson.map.exc.UnrecognizedPropertyException
    .from(UnrecognizedPropertyException.java:53)

Jackson正在抱怨,因为它无法在类Wrapper中找到一个名为“Wrapper”的字段。它这样做是因为JSON对象有一个名为“包装器”的属性。

我认为解决办法是将Wrapper类的字段重命名为“Wrapper”而不是“students”。


第一个答案几乎是正确的,但需要的是改变getter方法,不是字段——字段是私有的(不自动检测);此外,如果字段和getter都是可见的,则getter的优先级高于字段。(也有一些方法可以让私有字段可见,但如果你想要getter,那就没什么意义了)

所以getter应该被命名为getWrapper(),或者用以下注解:

@JsonProperty("wrapper")

如果你喜欢getter方法名。


你可以使用Jackson的类级注释:

import com.fasterxml.jackson.annotation.JsonIgnoreProperties

@JsonIgnoreProperties
class { ... }

它将忽略POJO中未定义的所有属性。当您只是在JSON中寻找几个属性而不想编写整个映射时,这非常有用。更多信息请访问杰克逊的网站。如果你想忽略任何未声明的属性,你应该这样写:

@JsonIgnoreProperties(ignoreUnknown = true)

问题是你的属性在你的JSON被称为“包装”和你的属性在wrapper .class被称为“学生”。

所以要么…

更正类或JSON中的属性名称。 根据StaxMan的注释注释您的属性变量。 注释setter(如果有的话)


我已经尝试了下面的方法,它适用于这样的JSON格式读取杰克逊。 使用已经建议的解决方案:用@JsonProperty("wrapper")注释getter

你的包装类

public Class Wrapper{ 
  private List<Student> students;
  //getters & setters here 
} 

我对包装类的建议

public Class Wrapper{ 

  private StudentHelper students; 

  //getters & setters here 
  // Annotate getter
  @JsonProperty("wrapper")
  StudentHelper getStudents() {
    return students;
  }  
} 


public class StudentHelper {

  @JsonProperty("Student")
  public List<Student> students; 

  //CTOR, getters and setters
  //NOTE: If students is private annotate getter with the annotation @JsonProperty("Student")
}

然而,这将为您提供格式的输出:

{"wrapper":{"student":[{"id":13,"name":Fred}]}}

更多信息请访问https://github.com/FasterXML/jackson-annotations


你可以使用

ObjectMapper objectMapper = getObjectMapper();
objectMapper.configure(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES, false);

它将忽略所有未声明的属性。


这个解决方案在读取json流时是通用的,只需要获取一些字段,而在域类中没有正确映射的字段可以忽略:

import org.codehaus.jackson.annotate.JsonIgnoreProperties;
@JsonIgnoreProperties(ignoreUnknown = true)

一个详细的解决方案是使用jsonschema2pojo等工具来自动生成所需的域类,例如从json响应的Schema中生成Student。您可以通过任何在线json到模式转换器来实现后者。


这对我来说非常有效

ObjectMapper objectMapper = new ObjectMapper();
objectMapper.configure(
    DeserializationConfig.Feature.FAIL_ON_UNKNOWN_PROPERTIES, false);

@JsonIgnoreProperties(ignoreUnknown = true)注释没有。


这比所有请参考此属性的工作更好。

import com.fasterxml.jackson.databind.DeserializationFeature;
import com.fasterxml.jackson.databind.ObjectMapper;

    ObjectMapper objectMapper = new ObjectMapper();
    objectMapper.configure(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES, false);
    projectVO = objectMapper.readValue(yourjsonstring, Test.class);

如果您正在使用Jackson 2.0

ObjectMapper mapper = new ObjectMapper();
mapper.disable(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES);

POJO应该定义为

响应类

public class Response {
    private List<Wrapper> wrappers;
    // getter and setter
}

包装器类

public class Wrapper {
    private String id;
    private String name;
    // getters and setters
}

和mapper来读取值

Response response = mapper.readValue(jsonStr , Response.class);

对我有用的,是把财产公诸于众。


要么改变

public Class Wrapper {
    private List<Student> students;
    //getters & setters here
}

to

public Class Wrapper {
    private List<Student> wrapper;
    //getters & setters here
}

----或----

将JSON字符串更改为

{"students":[{"id":"13","name":"Fred"}]}

对我来说,唯一的一条线

@JsonIgnoreProperties(ignoreUnknown = true)

也没起作用。

只需添加

@JsonInclude(Include.NON_EMPTY)

杰克逊测试盒框


这对我来说非常有效

objectMapper.configure(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES, false);

我通过简单地更改POJO类的setter和getter方法的签名来解决这个问题。我所要做的就是更改getObject方法,以匹配映射器正在寻找的内容。在我的情况下,我原来有一个getImageUrl,但JSON数据有image_url,这是抛出映射器。我把setter和getter都改成了getImage_url和setImage_url。


使用Jackson 2.6.0,这对我来说是有效的:

private static final ObjectMapper objectMapper = 
    new ObjectMapper()
        .configure(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES, false);

并带有设置:

@JsonIgnoreProperties(ignoreUnknown = true)

它为我工作了以下代码:

ObjectMapper mapper =new ObjectMapper();    
mapper.configure(DeserializationConfig.Feature.FAIL_ON_UNKNOWN_PROPERTIES, false);

你只需将List的字段从“students”更改为“wrapper”,只是json文件,mapper就会查找它。


根据文档,您可以忽略选定的字段或所有uknown字段:

 // to prevent specified fields from being serialized or deserialized
 // (i.e. not include in JSON output; or being set even if they were included)
 @JsonIgnoreProperties({ "internalId", "secretKey" })

 // To ignore any unknown properties in JSON input without exception:
 @JsonIgnoreProperties(ignoreUnknown=true)

您的输入

{"wrapper":[{"id":"13","name":"Fred"}]}

表示它是一个对象,具有一个名为“wrapper”的字段,它是一个学生的集合。所以我的建议是,

Wrapper = mapper.readValue(jsonStr , Wrapper.class);

其中Wrapper定义为

class Wrapper {
    List<Student> wrapper;
}

在我的例子中,它很简单:更新了rest -服务JSON对象(添加了一个属性),但没有更新rest -客户端JSON对象。只要我已经更新了JSON客户端对象的“不可识别的字段…’这个例外已经消失了。


您的json字符串没有与映射的类内联。 更改输入字符串

String jsonStr = "{\"students\"\:[{\"id\":\"13\",\"name\":\"Fred\"}]}";

或者更改映射的类

public class Wrapper {
    private List<Student> wrapper;
    //getters & setters here
}

新的Firebase Android引入了一些巨大的变化;文档副本下面:

[https://firebase.google.com/support/guides/firebase-android]:

更新您的Java模型对象

和2一样。x SDK, Firebase Database会自动将传递给DatabaseReference.setValue()的Java对象转换为JSON,并可以使用DataSnapshot.getValue()将JSON读入Java对象。

在新的SDK中,当使用DataSnapshot.getValue()将JSON读入Java对象时,JSON中的未知属性现在默认被忽略,因此您不再需要@JsonIgnoreExtraProperties(ignoreUnknown=true)。

为了在将Java对象写入JSON时排除字段/getter,注释现在被称为@Exclude而不是@JsonIgnore。

BEFORE

@JsonIgnoreExtraProperties(ignoreUnknown=true)
public class ChatMessage {
   public String name;
   public String message;
   @JsonIgnore
   public String ignoreThisField;
}

dataSnapshot.getValue(ChatMessage.class)

AFTER

public class ChatMessage {
   public String name;
   public String message;
   @Exclude
   public String ignoreThisField;
}

dataSnapshot.getValue(ChatMessage.class)

如果JSON中有一个Java类之外的额外属性,你会在日志文件中看到这样的警告:

W/ClassMapper: No setter/field for ignoreThisProperty found on class com.firebase.migrationguide.ChatMessage

您可以通过在类上添加@IgnoreExtraProperties注释来消除此警告。如果你想让Firebase数据库的行为,因为它在2。如果有未知的属性,你可以在类上放一个@ThrowOnExtraProperties注释。


由于json属性和java属性的名称不匹配,请将student字段注释如下

public Class Wrapper {
    @JsonProperty("wrapper")
    private List<Student> students;
    //getters & setters here
}

可以通过两种方式实现:

将POJO标记为忽略未知属性 @JsonIgnoreProperties(ignoreUnknown = true) 配置ObjectMapper序列化/反序列化POJO/json,如下所示: ObjectMapper mapper =new ObjectMapper(); // Jackson版本1。X mapper.configure (DeserializationConfig.Feature。FAIL_ON_UNKNOWN_PROPERTIES、假); // Jackson版本2。X mapper.configure (DeserializationFeature。FAIL_ON_UNKNOWN_PROPERTIES假)


您需要验证正在解析的类的所有字段,使其与原始JSONObject中的字段相同。它帮助了我,对我来说。

@JsonIgnoreProperties(ignoreUnknown = true)根本没有帮助。


这可能是一个非常晚的响应,但只是将POJO更改为这个应该解决问题中提供的json字符串(因为,输入字符串不像你说的那样在你的控制范围内):

public class Wrapper {
    private List<Student> wrapper;
    //getters & setters here
}

谷歌带我来这里,我很惊讶地看到答案…所有人都建议绕过这个错误(这个错误在发展过程中总是会反咬4倍),而不是解决它,直到这位先生恢复了对SO的信心!

objectMapper.readValue(responseBody, TargetClass.class)

用于将json String转换为类对象,缺少的是TargetClass应该有公共getter / setter。OP的问题片段中也缺少相同的内容!:)

通过龙目岛,你的类如下应该工作!!

@Data
@Builder
public class TargetClass {
    private String a;
}

在我的情况下,错误是由于以下原因

最初它工作得很好,然后我重命名了一个变量,使 代码的变化,它给了我这个错误。 然后我申请杰克逊无知财产也,但它没有工作。 最后,在重新定义我的getter和setter方法根据 我的变量名称此错误已解决

所以一定要重定义getter和setter。


将Wrapper类更改为

public Class Wrapper {

          @JsonProperty("wrapper")  // add this line
          private List<Student> students;
}

这样做的目的是将students字段识别为json对象的包装键。

此外,我个人更喜欢使用Lombok注解的getter和setter作为

@Getter
@Setter
public Class Wrapper {

          @JsonProperty("wrapper")  // add this line
          private List<Student> students;
}

因为我没有同时用Lombok和@JsonProperty测试上面的代码,所以我建议您也将下面的代码添加到Wrapper类中,以覆盖Lombok的默认getter和setter。

public List<Student> getWrapper(){
     return students;
}

public void setWrapper(List<Student> students){
     this.students = students;
}

还可以使用Jackson来反序列化列表。


将类字段设置为public而不是private。

public Class Student { 
    public String name;
    public String id;
    //getters & setters for name & id here
}

另一种可能是application.properties中的这个属性 Spring.jackson.deserialization.fail-on-unknown-properties =false,它不需要在应用程序中更改任何其他代码。当你认为合同是稳定的,你可以删除这个属性或标记它为真。


这可能不是OP遇到的相同问题,但如果有人带着和我同样的错误来到这里,那么这将帮助他们解决问题。当我使用来自不同依赖项的ObjectMapper作为JsonProperty注释时,我得到了与OP相同的错误。

如此:

import com.fasterxml.jackson.databind.ObjectMapper;
import com.fasterxml.jackson.annotation.JsonProperty;

不工作:

import org.codehaus.jackson.map.ObjectMapper; //org.codehaus.jackson:jackson-mapper-asl:1.8.8
import com.fasterxml.jackson.annotation.JsonProperty; //com.fasterxml.jackson.core:jackson-databind:2.2.3

进口com.fasterxml.jackson.annotation.JsonIgnoreProperties;

@JsonIgnoreProperties


在我的情况下,我必须添加公共getter和setter,以使字段私有。

ObjectMapper mapper = new ObjectMapper();
Application application = mapper.readValue(input, Application.class);

我使用jackson-databind 2.10.0。pr3。


当我们生成getter和setter,特别是以'is'关键字开头的getter和setter时,IDE通常会删除'is'。如。

private boolean isActive;

public void setActive(boolean active) {
   isActive = active;
}

public isActive(){
   return isActive;
}

在我的情况下,我只是改变了getter和setter。

private boolean isActive;

public void setIsActive(boolean active) {
   isActive = active;
}

public getIsActive(){
   return isActive;
}

它能够识别磁场。


FAIL_ON_UNKNOWN_PROPERTIES选项默认为true:

FAIL_ON_UNKNOWN_PROPERTIES (default: true)
Used to control whether encountering of unknown properties (one for which there is no setter; and there is no fallback "any setter" method defined using @JsonAnySetter annotation) should result in a JsonMappingException (when enabled), or just quietly ignored (when disabled)

添加setter和getter解决了这个问题,我觉得实际的问题是如何解决它,而不是如何抑制/忽略错误。 我得到了错误“无法识别的领域..没有被标记为可忽略的…”

虽然我在类的顶部使用了下面的注释,但它无法解析json对象并给我输入

@JsonIgnoreProperties(ignoreUnknown = true)

然后我意识到我没有添加setter和getter,在添加setter和getter到“包装器”和“学生”后,它就像一个魅力。


没有setter/getter的最短解决方案是将@JsonProperty添加到类字段:

public class Wrapper {
    @JsonProperty
    private List<Student> wrapper;
}

public class Student {
    @JsonProperty
    private String name;
    @JsonProperty
    private String id;
}

此外,您在json中称学生列表为“wrapper”,因此Jackson希望类具有一个名为“wrapper”的字段。


不知怎的,10年过去了,45篇帖子之后,还没有人对我的情况给出正确答案。

@Data //Lombok
public class MyClass {
    private int foo;
    private int bar;

    @JsonIgnore
    public int getFoobar() {
      return foo + bar;
    }
}

在我的例子中,我们有一个名为getFoobar()的方法,但没有foobar属性(因为它是从其他属性计算的)。类上的@JsonIgnoreProperties不起作用。

解决方案是用@JsonIgnore注释方法


如果由于某种原因,你不能将@JsonIgnoreProperties注释添加到你的类中,并且你是在一个web服务器/容器中,比如Jetty。您可以在自定义提供程序中创建和定制ObjectMapper

import javax.ws.rs.ext.ContextResolver;
import javax.ws.rs.ext.Provider;

import com.fasterxml.jackson.annotation.JsonInclude.Include;
import com.fasterxml.jackson.databind.DeserializationFeature;
import com.fasterxml.jackson.databind.ObjectMapper;

@Provider
public class CustomObjectMapperProvider implements ContextResolver<ObjectMapper> {

    private ObjectMapper objectMapper;

    @Override
    public ObjectMapper getContext(final Class<?> cls) {
        return getObjectMapper();
    }

    private synchronized ObjectMapper getObjectMapper() {
        if(objectMapper == null) {
            objectMapper = new ObjectMapper();
            objectMapper.configure(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES, false);
        }
        return objectMapper;
    }
}

如果你想将@JsonIgnoreProperties应用到应用程序中的所有类,那么最好的方法是重写Spring引导默认jackson对象。

在您的应用程序配置文件中定义一个bean来创建这样的杰克逊对象映射器。

@Bean
    public ObjectMapper getObjectMapper() {
        ObjectMapper mapper = new ObjectMapper();
        mapper.configure(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES, false);
        return mapper;
    }

现在,您不需要标记每个类,它将忽略所有未知属性。


为了防止其他人像我一样使用force-rest-api,下面是我如何使用这个讨论来解决它(Kotlin):

var result = forceApi.getSObject("Account", "idhere")
result.jsonMapper.configure( DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES,  false)
val account: Account = result.`as`(Account::class.java)

看起来force-rest-api使用的是jackson的旧版本。


根据这个文档,你可以使用Jackson2ObjectMapperBuilder来构建你的ObjectMapper:

@Autowired
Jackson2ObjectMapperBuilder objectBuilder;

ObjectMapper mapper = objectBuilder.build();
String json = "{\"id\": 1001}";

默认情况下,Jackson2ObjectMapperBuilder禁用错误unrecognizedpropertyexception。


Json:

 "blog_host_url": "some.site.com"

科特林字段

var blogHostUrl: String = "https://google.com"

在我的情况下,我只需要使用@JsonProperty注释在我的数据类。

例子:

data class DataBlogModel(
       @JsonProperty("blog_host_url") var blogHostUrl: String = "https://google.com"
    )

这是文章:https://www.baeldung.com/jackson-name-of-property


ObjectMapper objectMapper = new ObjectMapper()
.configure(DeserializationFeature.ACCEPT_EMPTY_ARRAY_AS_NULL_OBJECT, true);

我曾经遇到过这种情况,当时我的JSON有效负载包括一个API无法识别的属性。解决方案是重命名/删除违规属性。


我也遇到了类似的问题,唯一的区别是我正在使用YAML文件而不是JSON文件。所以我在创建ObjectMapper时使用YamlFactory()。另外,我的Java POJO的属性是私有的。 然而,所有这些答案中提到的解决方案都没有帮助。我的意思是,它停止抛出错误并开始返回Java对象但Java对象的属性将为null。

我只是做了以下几点:-

objectMapper.setVisibility(PropertyAccessor.FIELD, Visibility.ANY);

而不需要:-

objectMapper.configure(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES, false);

现在,我能够反序列化数据从Yaml文件到Java对象。所有属性现在都不是空的。这种解决方案也适用于JSON方法。


下面是我试图抑制未知的道具,不想解析。 共享Kotlin代码

val myResponse = jacksonObjectMapper().configure(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES, false)
            .readValue(serverResponse, FooResponse::class.java)

Java 11及更新版本的简单解决方案:

var mapper = new ObjectMapper()
        .registerModule(new JavaTimeModule())
        .disable(FAIL_ON_UNKNOWN_PROPERTIES)
        .disable(WRITE_DATES_AS_TIMESTAMPS)
        .enable(ACCEPT_EMPTY_ARRAY_AS_NULL_OBJECT);

重要的忽略是禁用'FAIL_ON_UNKNOWN_PROPERTIES'