详解jackson注解(三)jackson包含属性、忽略属性的注解

详解jackson注解(三)jackson包含属性、忽略属性的注解

jackson中,指定包含哪些属性、忽略哪些属性的注解:

注解类 描述
1 JsonIgnoreProperties 用于标记忽略一个或多个属性。可以注解在类上、构造函数、方法、字段上。
2 JsonIgnore @JsonIgnore注解用于在字段级别标记要忽略的属性。注意:系列化和反系列化时都会被忽略。
3 JsonInclude 使用@JsonInclude可以只包含非空的属性,也即排除值为empty、null的属性。
4 JsonAutoDetect 默认情况下,jackson获取public权限的字段进行系列化和反系列化。如果没有public修饰的字段,就会去获取public修饰的getter/setter。使用 JsonAutoDetect注解,我们就可以修改默认的行为。

一、JsonIgnoreProperties

用于标记忽略一个或多个属性。可以注解在类上、构造函数、方法、字段上。

import com.fasterxml.jackson.annotation.JsonIgnoreProperties;

import javax.persistence.JoinColumn;
import javax.persistence.ManyToOne;
import javax.persistence.OneToMany;
import java.io.Serializable;
import java.util.HashSet;
import java.util.Set;

/**
 * @author chushiyan
 * @email  [email protected]
 * @description 自关联的权限表。一个权限有一个父权限、多个子权限
 */
@Data
public class Permission implements Serializable {

    private String id;

    private String name;

    @ManyToOne
    @JoinColumn(name = "permission_parent_id", referencedColumnName = "permission_id")
    private Permission parent;

    // 必须使用JsonIgnoreProperties忽略parent属性
    // 否则查询到子权限,子权限有parent属性,又去查父权限,会导致系列化时出现死循环
    @JsonIgnoreProperties(value = "parent")
    @OneToMany(mappedBy = "parent")
    private Set<Permission> children = new HashSet<Permission>();
   
}

二、JsonIgnore

@JsonIgnore注解用于在字段级别标记要忽略的属性。注意:系列化和反系列化时都会被忽略。

1、bean

import com.fasterxml.jackson.annotation.JsonIgnore;
import lombok.Data;

import java.io.Serializable;

/**
 * @author chushiyan
 * @email  [email protected]
 * @description
 */
@Data
public class Article implements Serializable {

    private String id;

    private String name;

    @JsonIgnore
    private String category;
}

2、测试

 	@Test
    public void testJsonIgnore() throws JsonProcessingException {
        String json = "{"
                + "\"id\" : \"100\","
                + "\"name\" : \"详解jackson注解","
                + "\"category\" : \"jackson\""
                + "}";
        ObjectMapper mapper = new ObjectMapper();
        Article article = mapper.readValue(json,Article.class);
        System.out.println(article);

        Article article2 = new Article();
        article2.setId("100");
        article2.setName("详解jackson注解");
        article2.setCategory("jackson");
        String json2 = mapper.writeValueAsString(article2);
        System.out.println(json2);
}

3、控制台输出

Article(id=100, name=详解jackson注解, category=null)
{"id":"100","name":"详解jackson注解"}

反系列化时,json数据中明明包含了category值,但是最后Article对象的category属性值依然为null。就是因为使用了JsonIgnore。

三、JsonInclude

我们可以使用@JsonInclude排除某些empty、null、默认值的属性。

import com.fasterxml.jackson.annotation.JsonInclude;

/**
 * @author chushiyan
 * @email [email protected]
 * @description
 */
// 使用JsonInclude注解指明只包含非null的属性,即排除值为null的属性
@JsonInclude(JsonInclude.Include.NON_NULL)
public class Result {

    private boolean flag;// 是否成功

    private Integer code;// 返回码

    private String message;// 返回信息

    private Object data;// 返回数据

    public Result() {
    }

    public Result(boolean flag, Integer code, String message) {
        this.flag = flag;
        this.code = code;
        this.message = message;
    }

    public Result(boolean flag, Integer code, String message, Object data) {
        this.flag = flag;
        this.code = code;
        this.message = message;
        this.data = data;
    }

    public boolean isFlag() {
        return flag;
    }

    public void setFlag(boolean flag) {
        this.flag = flag;
    }

    public Integer getCode() {
        return code;
    }

    public void setCode(Integer code) {
        this.code = code;
    }

    public String getMessage() {
        return message;
    }

    public void setMessage(String message) {
        this.message = message;
    }

    public Object getData() {
        return data;
    }

    public void setData(Object data) {
        this.data = data;
    }
    
    @Override
    public String toString() {
        return "Result{" +
                "flag=" + flag +
                ", code=" + code +
                ", message='" + message + '\'' +
                ", data=" + data +
                '}';
    }

}

2、controller

 	@GetMapping("/test8")
    public Result testJsonInclude(){
        return new  Result(true,200,"查询成功");
    }

没有在Result类上加 @JsonInclude(JsonInclude.Include.NON_NULL) 注解时,返回的响应:

{
    "flag": true,
    "code": 200,
    "message": "查询成功",
    "data": null
}

加了之后的返回的响应:

{
    "flag": true,
    "code": 200,
    "message": "查询成功"
}

四、 JsonAutoDetect

默认情况下,jackson获取public权限的字段进行系列化和反系列化。如果没有public权限的字段,就会去获取public修饰的getter/setter。使用 JsonAutoDetect注解,我们就可以修改默认的行为。比如,下面的案例,即使字段都是private的、也没有getter/setter,也照样获取字段。

1、bean

首先,我们定义一个属性id/name都私有的而且没有getter/setter的一个bean

import com.fasterxml.jackson.annotation.JsonAutoDetect;

import java.io.Serializable;

/**
 * @author chushiyan
 * @email [email protected]
 * @description
 */
//@JsonAutoDetect(fieldVisibility = JsonAutoDetect.Visibility.ANY)
public class PrivateBean implements Serializable {

    private String id;

    private String name;

    public PrivateBean() {
    }

    public PrivateBean(String id, String name) {
        this.id = id;
        this.name = name;
    }

    @Override
    public String toString() {
        return "PrivateBean{" +
                "id='" + id + '\'' +
                ", name='" + name + '\'' +
                '}';
    }
}

2、反系列化测试:

 	@Test
    public void testJsonAutoDetect() throws JsonProcessingException {
        String json = "{"
                + "\"id\" : \"10\","
                + "\"name\" : \"chushiyan\""
                + "}";
        ObjectMapper mapper = new ObjectMapper();
        PrivateBean bean = mapper.readValue(json, PrivateBean.class);
        System.out.println(bean);
    }

报错:

com.fasterxml.jackson.databind.exc.UnrecognizedPropertyException: Unrecognized field "id" (class com.chushiyan.test.entity.PrivateBean), not marked as ignorable (0 known properties: ])
 at [Source: (String)"{"id" : "10","name" : "chushiyan"}"; line: 1, column: 10] (through reference chain: com.chushiyan.test.entity.PrivateBean["id"])

3、系列化测试

 	@Test
    public void testJsonAutoDetect2() throws JsonProcessingException {
        ObjectMapper mapper = new ObjectMapper();
        PrivateBean bean = new PrivateBean("11", "chushiyan");
        String json = mapper.writeValueAsString(bean);
        System.out.println(json);
    }

报错:

com.fasterxml.jackson.databind.exc.InvalidDefinitionException: No serializer found for class com.chushiyan.test.entity.PrivateBean and no properties discovered to create BeanSerializer (to avoid exception, disable SerializationFeature.FAIL_ON_EMPTY_BEANS)

4、bean上加上注解

import com.fasterxml.jackson.annotation.JsonAutoDetect;

import java.io.Serializable;

/**
 * @author chushiyan
 * @email [email protected]
 * @description
 */
@JsonAutoDetect(fieldVisibility = JsonAutoDetect.Visibility.ANY)
public class PrivateBean implements Serializable {

    private String id;

    private String name;

    public PrivateBean() {
    }

    public PrivateBean(String id, String name) {
        this.id = id;
        this.name = name;
    }

    @Override
    public String toString() {
        return "PrivateBean{" +
                "id='" + id + '\'' +
                ", name='" + name + '\'' +
                '}';
    }
}

5、再次分别测试反系列化、系列化

都运行正常,分别打印:

PrivateBean{id='10', name='chushiyan'}
{"id":"11","name":"chushiyan"}

你可能感兴趣的:(springboot)