Jackson 动态过滤属性

在把java对象转换为json时,有时需要依据需求动态过滤掉一些属性,比如使用注解形式过滤,这样是静态的。jackjson提供了这种动态过滤的。代码如下

 

import java.io.Writer;

import org.codehaus.jackson.JsonNode;
import org.codehaus.jackson.map.DeserializationConfig;
import org.codehaus.jackson.map.DeserializationConfig.Feature;
import org.codehaus.jackson.map.ObjectMapper;
import org.codehaus.jackson.map.annotate.JsonSerialize;
import org.codehaus.jackson.map.introspect.AnnotatedClass;
import org.codehaus.jackson.map.introspect.JacksonAnnotationIntrospector;
import org.codehaus.jackson.map.ser.FilterProvider;
import org.codehaus.jackson.map.ser.impl.SimpleBeanPropertyFilter;
import org.codehaus.jackson.map.ser.impl.SimpleFilterProvider;
import org.codehaus.jackson.type.JavaType;
import org.slf4j.Logger;

import com.maowu.commons.logutil.LogProxy;


public class JsonConverter
{
    private static final Logger log = LogProxy.getLogger(JsonConverter.class);

    public static final ObjectMapper mapper = new ObjectMapper();

    static
    {
        // 序列化时候,只序列化非空字段
        mapper.setSerializationConfig(mapper.getSerializationConfig().withSerializationInclusion(
                JsonSerialize.Inclusion.NON_NULL));
        // 当范序列化出现未定义字段时候,不出现错误
        DeserializationConfig deserializationConfig = mapper.getDeserializationConfig();
        deserializationConfig = deserializationConfig.without(Feature.FAIL_ON_UNKNOWN_PROPERTIES,
                Feature.FAIL_ON_NULL_FOR_PRIMITIVES);
        deserializationConfig = deserializationConfig.with(Feature.ACCEPT_EMPTY_STRING_AS_NULL_OBJECT,
                Feature.ACCEPT_SINGLE_VALUE_AS_ARRAY);

        mapper.setDeserializationConfig(deserializationConfig);
    }

    public static String format(Object obj)
    {
        try
        {
            return mapper.writeValueAsString(obj);
        }
        catch (Exception e)
        {
            log.debug("format obj failure.obj=" + obj);
            throw new RuntimeException("JsonUtil.format error:" + obj, e);
        }
    }

    public static void outputToWriter(Writer out, Object value)
    {
        try
        {
            mapper.writeValue(out, value);
        }
        catch (Exception e)
        {
            log.error("JsonUtil.outputToWriter error:" + value, e);
            throw new RuntimeException("JsonUtil.outputToWriter error:" + value, e);
        }
    }

    public static <T> T parse(JsonNode body, Class<T> clz)
    {
        try
        {
            return mapper.readValue(body, clz);
        }
        catch (Exception e)
        {
            log.error("JsonUtil.parse [" + clz + "]:" + body, e);
            throw new RuntimeException("JsonUtil.parse [" + clz + "]:" + body, e);
        }
    }

    public static <T> T parse(String str, Class<T> clz)
    {
        try
        {
            return mapper.readValue(str == null ? "{}" : str, clz);
        }
        catch (Exception e)
        {
            log.error("JsonUtil.parse [" + clz + "]:" + str, e);
            throw new RuntimeException("JsonUtil.parse [" + clz + "]:" + str, e);
        }
    }

    public static <T> T parseList(String str, Class<?> clz, Class<?> type)
    {
        try
        {
            JavaType javaType = mapper.getTypeFactory().constructParametricType(clz, type);
            return mapper.readValue(str, javaType);
        }
        catch (Exception e)
        {
            log.error("JsonUtil.parse [" + clz + "]:" + str, e);
            throw new RuntimeException("JsonUtil.parse [" + clz + "]:" + str, e);
        }
    }

    public static JsonNode tree(Object obj)
    {
        try
        {
            return mapper.valueToTree(obj);
        }
        catch (Exception e)
        {
            log.debug("format obj failure.obj=" + obj);
            throw new RuntimeException("JsonUtil.format error:" + obj, e);
        }
    }

    public static String serializeAllExcept(Object obj, String... filterFields)
    {
        try
        {
            ObjectMapper mapper = new ObjectMapper();
            mapper.setSerializationConfig(mapper.getSerializationConfig().withSerializationInclusion(
                    JsonSerialize.Inclusion.NON_NULL));

            FilterProvider filters = new SimpleFilterProvider().addFilter(obj.getClass().getName(),
                    SimpleBeanPropertyFilter.serializeAllExcept(filterFields));
            mapper.setFilters(filters);

            mapper.setAnnotationIntrospector(new JacksonAnnotationIntrospector()
            {
                @Override
                public Object findFilterId(AnnotatedClass ac)
                {
                    return ac.getName();
                }
            });

            return mapper.writeValueAsString(obj);
        }
        catch (Exception e)
        {
            log.debug("format obj failure.obj=" + obj);
            throw new RuntimeException("Json.format error:" + obj, e);
        }
    }

    public static String filterOutAllExcept(Object obj, String... filterFields)
    {
        try
        {
            ObjectMapper mapper = new ObjectMapper();
            mapper.setSerializationConfig(mapper.getSerializationConfig().withSerializationInclusion(
                    JsonSerialize.Inclusion.NON_NULL));

            FilterProvider filters = new SimpleFilterProvider().addFilter(obj.getClass().getName(),
                    SimpleBeanPropertyFilter.filterOutAllExcept(filterFields));
            mapper.setFilters(filters);

            mapper.setAnnotationIntrospector(new JacksonAnnotationIntrospector()
            {
                @Override
                public Object findFilterId(AnnotatedClass ac)
                {
                    return ac.getName();
                }
            });

            return mapper.writeValueAsString(obj);
        }
        catch (Exception e)
        {
            log.debug("format obj failure.obj=" + obj);
            throw new RuntimeException("Json.format error:" + obj, e);
        }
    }

}

`Jackson PropertyFilter` 是一个用于过滤 Jackson 序列化和反序列化过程中属性的接口。它允许您根据自定义逻辑决定要包含或排除哪些属性。 要使用 `Jackson PropertyFilter`,您需要实现 `com.fasterxml.jackson.databind.ser.PropertyFilter` 接口,并实现其中的 `include` 方法。在 `include` 方法中,您可以编写自定义逻辑来决定是否包含或排除属性。 下面是一个简单的示例,演示如何使用 `Jackson PropertyFilter`: ```java import com.fasterxml.jackson.core.JsonGenerator; import com.fasterxml.jackson.databind.JsonSerializer; import com.fasterxml.jackson.databind.SerializerProvider; import com.fasterxml.jackson.databind.ser.PropertyWriter; import com.fasterxml.jackson.databind.ser.impl.SimpleBeanPropertyFilter; import com.fasterxml.jackson.databind.ser.impl.SimpleFilterProvider; import com.fasterxml.jackson.databind.ObjectMapper; import java.io.IOException; public class PropertyFilterExample { public static class Person { private String name; private int age; // getters and setters } public static void main(String[] args) throws IOException { ObjectMapper objectMapper = new ObjectMapper(); // 创建一个自定义的 PropertyFilter SimpleBeanPropertyFilter propertyFilter = new SimpleBeanPropertyFilter() { @Override public void serializeAsField(Object pojo, JsonGenerator jgen, SerializerProvider provider, PropertyWriter writer) throws Exception { if (include(writer)) { writer.serializeAsField(pojo, jgen, provider); } } @Override public boolean include(PropertyWriter writer) { // 根据属性名称决定是否包含或排除属性 return !"age".equals(writer.getName()); } }; // 创建一个 FilterProvider 并注册自定义的 PropertyFilter SimpleFilterProvider filterProvider = new SimpleFilterProvider().addFilter("customFilter", propertyFilter); // 将 FilterProvider 设置到 ObjectMapper 中 objectMapper.setFilterProvider(filterProvider); // 序列化对象时,只会包含过滤器中指定的属性 Person person = new Person(); person.setName("John"); person.setAge(30); String json = objectMapper.writeValueAsString(person); System.out.println(json); } } ``` 在上面的示例中,我们创建了一个名为 `Person` 的简单类,并使用 `Jackson PropertyFilter` 过滤掉了属性 `age`。最终,我们将只获得包含 `name` 属性JSON 字符串。 请注意,这只是一个简单的示例,您可以根据自己的需求编写更复杂的逻辑来过滤属性。希望能帮助到你!如果你有其他问题,请随时提问。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值