jackson学习之七:常用Field注解

本篇概览

  1. 本文是《jackson学习》系列的第七篇,继续学习jackson强大的注解能力,本篇学习的是常用的Field注解,并通过实例来加深印象,下图是常用Field注解的简介:

在这里插入图片描述

  1. 接下来逐个学习;

不止是Filed

  • 虽然标题说是常用Field注解,其实上图中的这些注解也能用在方法上,只不过多数情况下这些注解修饰在field上更好理解一些,例如JsonIgnore,放在field上和get方法上都是可以的;

  • 接下来逐个学习;

源码下载

  1. 如果您不想编码,可以在GitHub下载所有源码,地址和链接信息如下表所示(https://github.com/zq2599/blog_demos):

| 名称 | 链接 | 备注 |

| :-- | :-- | :-- |

| 项目主页 | https://github.com/zq2599/blog_demos | 该项目在GitHub上的主页 |

| git仓库地址(https) | https://github.com/zq2599/blog_demos.git | 该项目源码的仓库地址,https协议 |

| git仓库地址(ssh) | git@github.com:zq2599/blog_demos.git | 该项目源码的仓库地址,ssh协议 |

  1. 这个git项目中有多个文件夹,本章的应用在jacksondemo文件夹下,如下图红框所示:

在这里插入图片描述

  1. jacksondemo是父子结构的工程,本篇的代码在annotation子工程中,里面的fieldannonation这个package下,如下图:

在这里插入图片描述

JsonProperty

  1. JsonProperty可以作用在成员变量和方法上,作用是在序列化和反序列化操作中指定json字段的名称;

  2. 先来看序列化操作(JsonPropertySerialization.java),如下所示,JsonProperty修饰了私有成员变量field0和公共方法getField1,并且field0没有get和set方法,是通过构造方法设置的,另外还要注意JsonProperty注解的index属性,用来指定序列化结果中的顺序,这里故意将field1的顺序设置得比field0靠前:

static class Test {

@JsonProperty(value=“json_field0”, index = 1)

private String field0;

@JsonProperty(value=“json_field1”, index = 0)

public String getField1() {

return “111”;

}

public Test(String field0) {

this.field0 = field0;

}

}

  1. 执行结果如下图红框所示,可见JsonProperty的value就是序列化后的属性名,另外带有JsonProperty注解的成员变量,即使是私有而且没有get和set方法,也能被成功序列化,而且顺序也和index属性对应:

在这里插入图片描述

  1. 接下来看反序列化操作(JsonPropertyDeserialization.java),注解相关代码如下,field0是私有且没有get和set方法,另外setField1方法也有JsonProperty注解:

static class Test {

@JsonProperty(value = “json_field0”)

private String field0;

private String field1;

@JsonProperty(value = “json_field1”)

public void setField1(String field1) {

this.field1 = field1;

}

@Override

public String toString() {

return “Test{” +

“field0='” + field0 + ‘’’ +

“, field1='” + field1 + ‘’’ +

‘}’;

}

}

  1. 用json字符串尝试反序列化,结果如下,可见field0和field1都能被正确赋值:

6.

JsonIgnore

  1. JsonIgnore好理解,作用在成员变量或者方法上,指定被注解的变量或者方法不参与序列化和反序列化操作;

  2. 先看序列化操作(JsonIgnoreSerialization.java),如下所示,Test类的field1字段和getField2方法都有JsonIgnore注解:

static class Test {

private String field0;

@JsonIgnore

private String field1;

private String field2;

public String getField0() { return field0; }

public void setField0(String field0) { this.field0 = field0; }

public String getField1() { return field1; }

public void setField1(String field1) { this.field1 = field1; }

public void setField2(String field2) { this.field2 = field2; }

@JsonIgnore

public String getField2() { return field2; }

}

  1. 给field0、field1、field2三个字段都赋值,再看序列化结果,如下图,可见field0和field2都被忽略了:

在这里插入图片描述

  1. 再来尝试JsonIgnore注解在反序列化场景的作用,注意反序列化的时候,JsonIgnore作用的方法应该是set了,如下图:

在这里插入图片描述

  1. 另外实测发现,反序列化的时候,JsonIgnore注解在get方法上也可以让对应字段被忽略;

JacksonInject

  1. JacksonInject的作用是在反序列化的时候,将配置好的值注入被JacksonInject注解的字段;

  2. 如下所示,Test类的field1和field2都有JacksonInject注解,不同的是field1指定了注入值的key为defaultField1,而field2由于没有指定key,只能按照类型注入:

static class Test {

private String field0;

@JacksonInject(value = “defaultField1”)

private String field1;

@JacksonInject

private String field2;

  1. 注入时所需的数据来自哪里呢?如下所示,通过代码配置的,可以指定key对应的注入值,也可以指定类型对应的注入值:

InjectableValues.Std injectableValues = new InjectableValues.Std();

// 指定key为"defaultField1"对应的注入参数

injectableValues.addValue(“defaultField1”,“field1 default value”);

// 指定String类型对应的注入参数

injectableValues.addValue(String.class,“String type default value”);

ObjectMapper mapper = new ObjectMapper(); // 把注入参数的配置设置给mapper

mapper.setInjectableValues(injectableValues);

  1. 反序列化结果如下图,可见field1和field2的值都是被注入的:

在这里插入图片描述

JsonSerialize

  1. JsonSerialize用于序列化场景,被此注解修饰的字段或者get方法会被用于序列化,并且using属性指定了执行序列化操作的类;

  2. 执行序列化操作的类,需要继承自JsonSerializer,如下所示,Date2LongSerialize的作用是将Date类型转成long类型:

static class Date2LongSerialize extends JsonSerializer {

@Override

public void serialize(Date value, JsonGenerator gen, SerializerProvider serializers) throws IOException {

gen.writeNumber(value.getTime());

}

}

  1. Test类的field0字段是私有的,且没有get和set方法,但是添加了注释JsonDeserialize就能被反序列化了,并且使用Date2LongSerialize类对将json中的long型转成field0所需的Date型:

static class Test {

@JsonDeserialize(using = Long2DateDeserialize.class)

private Date field0;

@Override

public String toString() { return “Test{” + “field0='” + field0 + ‘’’ + ‘}’; }

}

  1. 执行结果如下:

在这里插入图片描述

JsonDeserialize

  1. JsonDeserialize用于反序列化场景,被此注解修饰的字段或者set方法会被用于反序列化,并且using属性指定了执行反序列化操作的类;
<think>好的,我现在需要解决用户遇到的UnrecognizedPropertyException异常问题。用户提到在处理JSON反序列化时出现这个错误,具体是Jackson库报出无法识别的字段。我要根据提供的引用信息来整理解决方案。 首先,我得回忆一下这个异常通常是什么原因导致的。根据引用[5]中的描述,这个异常通常是因为JSON数据中有实体类未定义的字段。例如,如果JSON里有一个"dns"字段,但对应的Java类中没有这个属性,Jackson在反序列化时就会抛出这个异常。 接下来,我需要整理可能的解决方法。根据提供的引用,有几个解决方案被提到: 1. 使用@JsonIgnoreProperties注解忽略未知字段。引用[4]和[5]都提到了这个方法,通过在类上添加@JsonIgnoreProperties(ignoreUnknown = true)来让Jackson忽略未定义的字段。这应该是最直接的解决办法。 2. 配置ObjectMapper。引用[5]提到可以通过配置ObjectMapper的DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES为false来全局忽略未知字段。这可能适用于整个应用,而不用在每个类上加注解。 3. 更新实体类字段。如果JSON中的字段是预期的,应该更新实体类添加对应的属性,并确保命名正确或者使用@JsonProperty注解映射字段名。比如引用[2]中的"dnsMonitoring"字段,如果确实需要,就应该在类中定义这个属性。 此外,用户可能需要检查JSON结构和实体类是否一致。比如引用[3]中的"st"字段和引用[4]中的"safeKey"字段,都是因为JSON中存在而类中没有对应的属性。所以,确保实体类与JSON数据结构匹配也是关键。 需要注意,不同的解决方案有不同的适用场景。如果只是临时忽略未知字段,用注解或配置ObjectMapper比较合适;如果字段是需要的,就应该更新实体类。同时,用户可能遇到的问题可能涉及到字段名称不一致,比如大小写或下划线等,这时候@JsonProperty可以帮助映射。 最后,可能需要提醒用户检查依赖版本,确保使用的Jackson库没有问题,但根据引用中的版本信息(如2.13.5),可能已经是最新稳定版,所以版本问题可能性较低。 总结下来,解决方案应该分步骤,首先推荐使用注解或配置ObjectMapper来忽略未知字段,其次建议更新实体类,最后检查JSON结构和字段命名。需要将这些步骤清晰地呈现给用户,并提供代码示例,如添加注解或配置ObjectMapper的方式。</think>### 解决方案:`UnrecognizedPropertyException`异常处理 `UnrecognizedPropertyException`是Jackson库在JSON反序列化时遇到的常见错误,表示目标实体类中缺少与JSON字段对应的属性[^5]。以下是具体解决步骤: --- #### **1. 忽略未知字段** 在实体类上添加`@JsonIgnoreProperties`注解,声明忽略未定义的字段: ```java @JsonIgnoreProperties(ignoreUnknown = true) // 关键注解[^4][^5] public class MyEntity { private String name; // 其他已定义字段... } ``` --- #### **2. 全局配置ObjectMapper** 通过配置`ObjectMapper`禁用严格模式,全局忽略未知字段: ```java ObjectMapper mapper = new ObjectMapper(); mapper.configure(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES, false); // 关闭严格检查[^5] MyEntity entity = mapper.readValue(jsonString, MyEntity.class); ``` --- #### **3. 更新实体类字段** 如果JSON字段是预期需要的,应在实体类中补充对应属性: ```java public class MyEntity { private String name; // 新增缺失字段(需与JSON字段名一致) private String dnsMonitoring; // 例如解决引用[2]的问题 } ``` 若字段名不匹配,可使用`@JsonProperty`注解显式映射: ```java @JsonProperty("safeKey") // 映射JSON字段名 private String safeKey; ``` --- #### **4. 检查JSON与实体类一致性** 确保JSON数据结构与实体类定义完全匹配: - **字段命名**:检查大小写、下划线命名(如`dnsMonitoring`与`dns_monitoring`差异)[^2] - **嵌套结构**:若JSON包含嵌套对象,需定义对应的内部类结构 --- #### **5. 调试与日志** 启用Jackson的调试日志,定位具体未知字段: ```java mapper.enable(DeserializationFeature.FAIL_ON_IGNORED_PROPERTIES); mapper.enable(DeserializationFeature.FAIL_ON_INVALID_SUBTYPE); ``` --- ### 总结 | 方法 | 适用场景 | 优点 | 缺点 | |------|----------|------|------| | `@JsonIgnoreProperties` | 局部忽略特定类的未知字段 | 精准控制 | 需修改类定义 | | 全局配置`ObjectMapper` | 项目级忽略未知字段 | 一劳永逸 | 可能掩盖潜在问题 | | 更新实体类字段 | 需要完整解析JSON | 数据完整性高 | 需同步维护字段 | ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值