网站建设的域名一个网站推广
参考链接:https://www.baeldung.com/jackson-annotations
之前和第三方对接,返回的接口中的属性名称是拼音字母大写,奇怪,反序列化的时候好多字段都为空,没设置进去。
因为对接前,我先用 IntelliJ IDEA 的 Http Client 工具调试接口,返回的属性并不为空,但是用 RestTemplate 调用接口反序列化后的字段都为空。跟踪代码后,发现在收集反序列化后的对象的属性名称的时候,把大写字段名称都改写成了小写字母。而 json 字符串的名称都是大写的,但是在 Bean 的属性名称集合中的名称都是小写的,自然就匹配不上了。所以,反序列化的时候,大写字段的值都为空了。
下面用一段代码来复现一下 :
import com.fasterxml.jackson.databind.DeserializationFeature;
import com.fasterxml.jackson.databind.ObjectMapper;/*** @author shifengqiang 2023/2/22 16:20*/
public class TestBean {private int id;private String BH;public String getBH() {return BH;}public void setBH(String BH) {this.BH = BH;}public int getId() {return id;}public void setId(int id) {this.id = id;}@Overridepublic String toString() {return "{\"TestBean\":{"+ "\"id\":\"" + id + " \""+ ",\"BH\":\"" + BH + " \""+ "}}";}public static void main(String[] args) throws Exception {String json = "{\"id\":1,\"BH\":\"aaa\"}";ObjectMapper mapper = new ObjectMapper();mapper.disable(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES);TestBean test = mapper.readValue(json, TestBean.class);String json1 = mapper.writeValueAsString(test);System.out.println("序列化前的 json :");System.out.println(json);System.out.println();System.out.println("反序列化后的对象:");System.out.println(test);System.out.println();System.out.println("反序列化后的 bean 再序列化后的 json :");System.out.println(json1);System.out.println();}
}
下面是控制台的输出结果:
序列化前的 json :
{"id":1,"BH":"aaa"}反序列化后的对象:
{"TestBean":{"id":"1 ","BH":"null "}}反序列化后的 bean 再序列化后的 json :
{"id":1,"bh":null}
对 jackson 原理感兴趣的同学可以跟踪代码,读一下实现代码。
- 收集 bean 的属性名称 :
com.fasterxml.jackson.databind.introspect.POJOPropertiesCollector#collectAll() 方法中对 bean 的 field、get 方法、set 方法进行了收集。jackson 会忽略所有 private 字段、方法。最终收集到的 field 、get/set 方法名作为了 属性名称集合 。反序列化的时候,如果 json 中的属性名在属性名称集合中找不到的话,就没法设置值了。 - 把大写名称改为小写的具体实现方法 :
在 com.fasterxml.jackson.databind.util.BeanUtil#legacyManglePropertyName() 方法中把大写名称改为了小写。
/*** Method called to figure out name of the property, given * corresponding suggested name based on a method or field name.** @param basename Name of accessor/mutator method, not including prefix* ("get"/"is"/"set")*/protected static String legacyManglePropertyName(final String basename, final int offset){final int end = basename.length();if (end == offset) { // empty name, nopereturn null;}// next check: is the first character upper case? If not, return as ischar c = basename.charAt(offset);char d = Character.toLowerCase(c);if (c == d) {return basename.substring(offset);}// otherwise, lower case initial chars. Common case first, just one charStringBuilder sb = new StringBuilder(end - offset);sb.append(d);int i = offset+1;for (; i < end; ++i) {c = basename.charAt(i);d = Character.toLowerCase(c);if (c == d) {sb.append(basename, i, end);break;}sb.append(d);}return sb.toString();}
下面两种方法任意一种都可以解决大小写不匹配的问题 :
- 在 BH 字段上加注解 @JsonProperty(“BH”) 来显式的声音属性的名称是 BH 。
- 在 getBH() 方法或者 setBH() 这两个方法中的任意一个方法上加 @JsonProperty(“BH”) 注解。