jackson自定义策略

Jackson支持在处理数据的时候,使用不同于对象字段名的JSON名称(Jackson内部使用),来代替原来的字段名进行序列化和反序列化。

主要有几种实现方式:

  • 使用@JsonProperty指定固定的名称进行名称映射;
  • 使用预定义的命名策略PropertyNamingStrategy,设置全局或单个类的命名策略;
  • 扩展PropertyNamingStrategy,实现自定义命名策略,读和写支持使用不同的命名策略。
一.属性名称@JsonProperty

对于需要修改名称的字段,可以在字段或getter方法添加@JsonProperty注解,指定一个固定的名称来替代原来的字段名。

public class AnimalPropertyName {

    @JsonProperty("animalName") // 字段重命名. 可以对字段或getter进行声明
    private String name;
    private int sex;
    private Integer weight;

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    @JsonProperty("animalSex") // 字段重命名. 可以对字段或getter进行声明
    public int getSex() {
        return sex;
    }

    public void setSex(int sex) {
        this.sex = sex;
    }

    public Integer getWeight() {
        return weight;
    }

    public void setWeight(Integer weight) {
        this.weight = weight;
    }

    @Override
    public String toString() {
        return "Animal [name=" + name + ", sex=" + sex + ", weight=" + weight + "]";
    }

}
  • 使用@JsonProperty(“字段别名”)注解,序列化和反序列化时使用指定的名称替代字段名

       @Test
        public void propertyName() throws IOException, JsonProcessingException {
            AnimalPropertyName animal = new AnimalPropertyName();
            animal.setName("sam");
            animal.setSex(26);
            animal.setWeight(100);
    
            ObjectMapper mapper = new ObjectMapper();
    
            // 序列化
            String jsonString = mapper.writeValueAsString(animal);
            System.out.println(jsonString);
    
            // 正确反序列化
            String jsonString2 = "{\"weight\":200,\"animalName\":\"sam2\",\"animalSex\":2}";
            AnimalPropertyName animal2 = mapper.readValue(jsonString2, AnimalPropertyName.class);
            System.out.println(animal2.toString());
    
            // 错误反序列化. 不能使用原来的字段名name和sex,需要使用注解的名称animalName和animalSex
            String jsonString3 = "{\"weight\":200,\"name\":\"sam2\",\"sex\":2}";
            ;
            AnimalPropertyName animal3 = mapper.readValue(jsonString3, AnimalPropertyName.class);
            System.out.println(animal3.toString());
        }
    
  • 执行结果

    {"weight":100,"animalName":"sam","animalSex":26}
    Animal [name=sam2, sex=2, weight=200]
    
    
    com.fasterxml.jackson.databind.exc.UnrecognizedPropertyException: Unrecognized field "name" (class com.example.note.bean.AnimalPropertyName), not marked as ignorable (3 known properties: "weight", "animalSex", "animalName"])
     at [Source: (String)"{"weight":200,"name":"sam2","sex":2}"; line: 1, column: 23] (through reference chain: com.example.note.bean.AnimalPropertyName["name"])
    
    

可以看到,第一行的序列化和第二行的反序列化结果,符合我们的预期。

但第二个反序列化抛出了异常,提示无法识别name字段。

第一次和第二次反序列化的区别,在于第一次使用了映射之后的属性名,而第二次则使用了对象的字段名。

也就是说,如果指定了属性名称,那么在进行序列化和反序列化时,都需要使用指定的属性名称。

二 .预定义命名策略PropertyNamingStrategy

使用属性名称,可以为不同字段设置不同的名称,而且这个名称是固定的。

如果被操作的字段,都需要遵循相同的命名规则,那么可以使用命名策略PropertyNamingStrategy来简化命名操作。

全局的命名策略
Jackson为我们提供了6个默认的命名策略。

public class AnimalNaming {

	private String animalName;
	private int animalSex;
	private int animalWeight;
	
	public String getAnimalName() {
		return animalName;
	}
	
	public void setAnimalName(String animalName) {
		this.animalName = animalName;
	}
	
	public int getAnimalSex() {
		return animalSex;
	}
	
	public void setAnimalSex(int animalSex) {
		this.animalSex = animalSex;
	}
	
	public int getAnimalWeight() {
		return animalWeight;
	}
	
	public void setAnimalWeight(int animalWeight) {
		this.animalWeight = animalWeight;
	}
	
	@Override
	public String toString() {
		return "AnimalNaming [animalName=" + animalName + ", animalSex=" + animalSex + ", animalWeight=" + animalWeight
		        + "]";
	}

}
  @Test
    public void naming() throws IOException {
        AnimalNaming animal = new AnimalNaming();
        animal.setAnimalName("sam");
        animal.setAnimalSex(1);
        animal.setAnimalWeight(100);

        // 驼峰命名,字段的首字母小写. {"animalName":"sam","animalSex":1,"animalWeight":100}
        ObjectMapper mapper1 = new ObjectMapper();
        mapper1.setPropertyNamingStrategy(PropertyNamingStrategy.LOWER_CAMEL_CASE);
        System.out.println(mapper1.writeValueAsString(animal));

        // 驼峰命名,字段的首字母大写. {"AnimalName":"sam","AnimalSex":1,"AnimalWeight":100}
        ObjectMapper mapper2 = new ObjectMapper();
        mapper2.setPropertyNamingStrategy(PropertyNamingStrategy.UPPER_CAMEL_CASE);
        System.out.println(mapper2.writeValueAsString(animal));

        // 字段小写,多个单词以下划线_分隔. {"animal_name":"sam","animal_sex":1,"animal_weight":100}
        ObjectMapper mapper3 = new ObjectMapper();
        mapper3.setPropertyNamingStrategy(PropertyNamingStrategy.SNAKE_CASE);
        System.out.println(mapper3.writeValueAsString(animal));

        // 字段小写,多个单词以中横线-分隔. {"animal-name":"sam","animal-sex":1,"animal-weight":100}
        ObjectMapper mapper4 = new ObjectMapper();
        mapper4.setPropertyNamingStrategy(PropertyNamingStrategy.KEBAB_CASE);
        System.out.println(mapper4.writeValueAsString(animal));

        // 字段小写,多个单词间无分隔符. {"animalname":"sam","animalsex":1,"animalweight":100}
        ObjectMapper mapper5 = new ObjectMapper();
        mapper5.setPropertyNamingStrategy(PropertyNamingStrategy.LOWER_CASE);
        System.out.println(mapper5.writeValueAsString(animal));

        // 字段小写,多个单词以点号.分隔. {"animal.name":"sam","animal.sex":1,"animal.weight":100}
        ObjectMapper mapper6 = new ObjectMapper();
        mapper6.setPropertyNamingStrategy(PropertyNamingStrategy.LOWER_DOT_CASE);
        System.out.println(mapper6.writeValueAsString(animal));
    }
  • 执行结果:

    {"animalName":"sam","animalSex":1,"animalWeight":100}
    {"AnimalName":"sam","AnimalSex":1,"AnimalWeight":100}
    {"animal_name":"sam","animal_sex":1,"animal_weight":100}
    {"animal-name":"sam","animal-sex":1,"animal-weight":100}
    {"animalname":"sam","animalsex":1,"animalweight":100}
    {"animal.name":"sam","animal.sex":1,"animal.weight":100}
    

通过ObjectMapper指定命名策略,这个策略是全局的。也就是说,只要使用了该ObjectMapper,那么都会应用该命名策略。

单个类的命名策略
如果命名策略只需要作用于某个类,那么可以使用@JsonNaming注解。

@JsonNaming(PropertyNamingStrategy.SnakeCaseStrategy.class)
public class AnimalJSONNaming {

	private String animalName;
	private int animalSex;
	private int animalWeight;
	
	public String getAnimalName() {
		return animalName;
	}
	
	public void setAnimalName(String animalName) {
		this.animalName = animalName;
	}
	
	public int getAnimalSex() {
		return animalSex;
	}
	
	public void setAnimalSex(int animalSex) {
		this.animalSex = animalSex;
	}
	
	public int getAnimalWeight() {
		return animalWeight;
	}
	
	public void setAnimalWeight(int animalWeight) {
		this.animalWeight = animalWeight;
	}
	
	@Override
	public String toString() {
		return "AnimalNaming [animalName=" + animalName + ", animalSex=" + animalSex + ", animalWeight=" + animalWeight
		        + "]";
	}

}
/**

 * 使用@JsonNaming注解,指定属性命名策略

 * 

 * @throws IOException
   */
   @Test
   public void jsonNaming() throws IOException {
   AnimalJSONNaming animal = new AnimalJSONNaming();
   animal.setAnimalName("sam");
   animal.setAnimalSex(1);
   animal.setAnimalWeight(100);

   ObjectMapper mapper = new ObjectMapper();
   System.out.println(mapper.writeValueAsString(animal));
   }
  • 执行结果:

    {"animal_name":"sam","animal_sex":1,"animal_weight":100}
    
    三.自定义PropertyNamingStrategy

    看到源码可以发现预定义的策略都是继承了PropertyNamingStrategyBase这个类,于是我们自己实现他,实现一个字段大写,多个单词以下划线_分隔的策略.

    /**
     * @Auther: wxy
     * @Date: 2021/8/8 13:37
     * @Description:
     */
    public class MyStrategy extends PropertyNamingStrategy.PropertyNamingStrategyBase {
        @Override
        public String translate(String input) {
            if (input == null) return input; // garbage in, garbage out
            int length = input.length();
            StringBuilder result = new StringBuilder(length * 2);
            int resultLength = 0;
            boolean wasPrevTranslated = false;
            for (int i = 0; i < length; i++) {
                char c = input.charAt(i);
                if (i > 0 || c != '_') // skip first starting underscore
                {
                    if (Character.isUpperCase(c)) {
                        if (!wasPrevTranslated && resultLength > 0 && result.charAt(resultLength - 1) != '_') {
                            result.append('_');
                            resultLength++;
                        }
                        c = Character.toLowerCase(c);
                        wasPrevTranslated = true;
                    } else {
                        wasPrevTranslated = false;
                    }
                    result.append(c);
                    resultLength++;
                }
            }
            return resultLength > 0 ? result.toString().toUpperCase() : input.toUpperCase();
        }
    }
    
        @Test
        public void selfStrategy() throws IOException {
    
            AnimalNaming animal = new AnimalNaming();
            animal.setAnimalName("sam");
            animal.setAnimalSex(1);
            animal.setAnimalWeight(100);
            ObjectMapper objectMapper = new ObjectMapper();
            // 字段大写,多个单词以下划线_分隔. 	{"ANIMAL_NAME":"sam","ANIMAL_SEX":1,"ANIMAL_WEIGHT":100}
            objectMapper.setPropertyNamingStrategy(new MyStrategy());
            System.out.println(objectMapper.writeValueAsString(animal));
        }
    
  • 0
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
在Spring Boot中,可以通过自定义Jackson来实现自定义序列化和反序列化的需求。以下是一个简单的示例,演示了如何自定义Jackson。 首先,我们需要创建一个Jackson的ObjectMapper Bean,并指定我们自定义的序列化器和反序列化器。示例中,我们自定义了一个格式化日期的序列化器和反序列化器。 ```java @Configuration public class JacksonConfig { @Bean public ObjectMapper objectMapper() { ObjectMapper objectMapper = new ObjectMapper(); SimpleModule module = new SimpleModule(); module.addSerializer(Date.class, new CustomDateSerializer()); module.addDeserializer(Date.class, new CustomDateDeserializer()); objectMapper.registerModule(module); return objectMapper; } } ``` 然后,我们需要实现自定义的序列化器和反序列化器。示例中,我们自定义了一个格式化日期的序列化器和反序列化器。 ```java public class CustomDateSerializer extends JsonSerializer<Date> { private static final SimpleDateFormat dateFormat = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"); @Override public void serialize(Date value, JsonGenerator gen, SerializerProvider serializers) throws IOException { gen.writeString(dateFormat.format(value)); } } public class CustomDateDeserializer extends JsonDeserializer<Date> { private static final SimpleDateFormat dateFormat = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"); @Override public Date deserialize(JsonParser p, DeserializationContext ctxt) throws IOException { String dateValue = p.getText(); try { return dateFormat.parse(dateValue); } catch (ParseException e) { throw new RuntimeException(e); } } } ``` 现在,我们已经完成了自定义Jackson的配置。在使用时,我们可以直接使用@Autowired注解注入ObjectMapper Bean,或者使用@JsonComponent注解来标识我们的自定义序列化器和反序列化器。 ```java @RestController public class UserController { @Autowired private ObjectMapper objectMapper; @PostMapping("/user") public User addUser(@RequestBody User user) throws JsonProcessingException { String json = objectMapper.writeValueAsString(user); System.out.println(json); return user; } } @JsonComponent public class DateJsonComponent { public static class Serializer extends JsonSerializer<Date> { private static final SimpleDateFormat dateFormat = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"); @Override public void serialize(Date value, JsonGenerator gen, SerializerProvider serializers) throws IOException { gen.writeString(dateFormat.format(value)); } } public static class Deserializer extends JsonDeserializer<Date> { private static final SimpleDateFormat dateFormat = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"); @Override public Date deserialize(JsonParser p, DeserializationContext ctxt) throws IOException { String dateValue = p.getText(); try { return dateFormat.parse(dateValue); } catch (ParseException e) { throw new RuntimeException(e); } } } } ``` 以上就是自定义Jackson的简单示例。通过自定义Jackson,我们可以轻松实现自定义的序列化和反序列化需求。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值