一篇文章了解Jackson注解@JsonFormat及失效解决办法
背景
项目中使用WRITE_DATES_AS_TIMESTAMPS: true
转换日期格式为时间戳未生效。如下:
spring: jackson: time-zone: Asia/Shanghai serialization: WRITE_DATES_AS_TIMESTAMPS: true
尝试是否关于时间的注解是否会生效,使用@JsonForma
和@JsonFiled
均失效。
常见失效原因及解决方法
@JsonFormat是用于指定Java对象序列化为JSON字符串时的格式化方式的注解。如果@JsonFormat失效,可能是以下原因:
未正确导入Jackson库
注解位置不正确
注解参数设置错误
解决方法:
- 确认是否正确导入了Jackson库,可以在pom.xml文件中添加以下依赖:
<dependency> <groupId>com.fasterxml.jackson.core</groupId> <artifactId>jackson-databind</artifactId> <version>2.12.3</version> </dependency>
2、确认注解位置是否正确,例如:
public class User { @JsonFormat(pattern = "yyyy-MM-dd") private Date birthday; // ... }
3、确认注解参数是否正确设置,例如:
@JsonFormat(shape = JsonFormat.Shape.STRING, pattern = "yyyy-MM-dd'T'HH:mm:ss.SSSZ") public Date getCreatedDate() { return createdDate; }
以上是常见的解决方法,如果还有问题可以提供更具体的错误信息进行排查。
本次问题解决
失效原因
因为项目中使用了Gson替换Jackson。代码如下:
import com.google.gson.JsonElement; import com.google.gson.JsonParser; import com.google.gson.JsonSerializationContext; import com.google.gson.JsonSerializer; import springfox.documentation.spring.web.json.Json; import java.lang.reflect.Type; public class JsonToGson implements JsonSerializer<Json> { @Override public JsonElement serialize(Json json, Type type, JsonSerializationContext context) { return JsonParser.parseString(json.value()); } }
@Configuration public class SwaggerWebConfiguration implements WebMvcConfigurer { @Override public void addResourceHandlers(ResourceHandlerRegistry registry) { registry.addResourceHandler("/swagger-ui/**") .addResourceLocations("classpath:/META-INF/resources/webjars/springfox-swagger-ui/"); } @Override public void configureMessageConverters(List<HttpMessageConverter<?>> converters) { converters.removeIf(httpMessageConverter -> httpMessageConverter instanceof MappingJackson2HttpMessageConverter); converters.add(gsonHttpMessageConverters()); } @Bean public GsonHttpMessageConverter gsonHttpMessageConverters() { Gson gson = new GsonBuilder() .registerTypeAdapter(Json.class, new SpringfoxJsonToGsonAdapter()) .create(); GsonHttpMessageConverter gsonConverter = new GsonHttpMessageConverter(gson); return gsonConverter; } }
解决方案一:去掉Gson转换即可。
解决方案二:重新添加MappingJackson2HttpMessageConverter的bean
@Autowired(required = false) private MappingJackson2HttpMessageConverter mappingJackson2HttpMessageConverter; @Override public void extendMessageConverters(List<HttpMessageConverter<?>> converters) { converters.removeIf(converter -> converter instanceof MappingJackson2HttpMessageConverter); if (Objects.isNull(mappingJackson2HttpMessageConverter)) { converters.add(0, new MappingJackson2HttpMessageConverter()); } else { converters.add(0, mappingJackson2HttpMessageConverter); } }
或
@Autowired(required = false) private MappingJackson2HttpMessageConverter mappingJackson2HttpMessageConverter; @Override public void configureMessageConverters(List<HttpMessageConverter<?>> converters) { converters.removeIf(converter -> converter instanceof MappingJackson2HttpMessageConverter); if (Objects.isNull(mappingJackson2HttpMessageConverter)) { converters.add(0, new MappingJackson2HttpMessageConverter()); } else { converters.add(0, mappingJackson2HttpMessageConverter); } }
解析
Jackson一直是springframework默认的json库,从4.1开始,springframework支持通过配置GsonHttpMessageConverter的方式使用Gson。
在Spring MVC
中,一旦请求退出@Controller
,它将寻找一个视图来呈现。当指定了@RequestBody
或@RestControlle
r时,我们会告诉Spring跳过这一步,将java对象通过model写入响应结果。Spring通过HttpMessageConverter
来执行Java对象向其它类型(通常是Json)的转换,Spring默认使用的是 MappingJackson2HttpMessageConverter
,所以如果希望使用Gson来执行这种转换,可用使用GsonHttpMessageConverter
替换Jackson
。
小拓展
在Spring Boot提供了与三个JSON映射库的集成:Gson、默认库 Jackson 、JSON-B
@JsonFormat与@JSONField简介
@JsonFormat
@JsonFormat是Java中的一个Jackson注解,用于指定某个字段或属性,或整个类的JSON序列化或反序列化格式。它允许在将数据类型转换为或从JSON格式转换时自定义日期、数字、布尔值等数据类型的格式。
例如,如果您有一个Java类,其中包含一个Date字段,您希望以特定格式将其序列化为JSON,您可以使用@JsonFormat注解来指定该格式:
示例如下:Jackson在序列化或反序列化myDate字段时使用“yyyy-MM-dd”格式。
public class MyClass { @JsonFormat(pattern="yyyy-MM-dd") private Date myDate; // ... }
@JSONField
@JSONField是阿里巴巴的fastjson库中的一个注解,用于指定Java对象属性在序列化为JSON字符串时的名称、顺序、格式等信息。例如,可以使用@JSONField(name=“username”)指定Java对象属性在序列化为JSON字符串时使用"username"作为属性名。
区别
@JsonFormat和@JSONField都是用于控制Java对象属性在序列化为JSON字符串时的格式,但是它们有一些不同之处。
@JsonFormat是Jackson库中的一个注解,用于指定Java对象属性在序列化为JSON字符串时的日期、时间、数字等格式。例如,可以使用@JsonFormat(pattern=“yyyy-MM-dd HH:mm:ss”)指定Java对象属性在序列化为JSON字符串时使用指定的日期时间格式。
@JSONField是阿里巴巴的fastjson库中的一个注解,用于指定Java对象属性在序列化为JSON字符串时的名称、顺序、格式等信息。例如,可以使用@JSONField(name=“username”)指定Java对象属性在序列化为JSON字符串时使用"username"作为属性名。
因此,虽然@JsonFormat和@JSONField都可以用于控制Java对象属性在序列化为JSON字符串时的格式,但是它们的作用范围和使用方式略有不同。
jackson自定义日期注解
依赖HttpMessageConverter默认实现如下:
jackson:MappingJackson2HttpMessageConverter;
gson:GsonHttpMessageConverter;
fastjson : FastJsonHttpMessageConverter
1、自定义转换类:
public class DateToJsonSerializer extends JsonSerializer<Date> implements ContextualSerializer { private DateFormat df = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"); public DateToJsonSerializer() { } @Override public void serialize(Date arg0, JsonGenerator arg1, SerializerProvider arg2) throws IOException { if (null != arg0) { arg1.writeString(this.df.format(arg0)); } } @Override public JsonSerializer<?> createContextual(SerializerProvider prov, BeanProperty property) { AnnotationMap annotated = property.getMember().getAllAnnotations(); JsonFormat jsonFormat = annotated.get(JsonFormat.class); if (jsonFormat != null && jsonFormat.pattern() != null){ df = new SimpleDateFormat(jsonFormat.pattern()); } return this; } }
2、使用
public class MyClass { @JsonSerialize(using = DateToJsonSerializer .class) private Date myDate; // ... }
总结
到此这篇关于Jackson注解@JsonFormat及失效解决办法的文章就介绍到这了,更多相关Jackson注解@JsonFormat及失效解决内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!
相关文章
springboot项目中mybatis-plus@Mapper注入失败问题
这篇文章主要介绍了springboot项目中mybatis-plus@Mapper注入失败问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教2024-07-07jmeter添加自定函数的实例(jmeter5.3+IntelliJ IDEA)
这篇文章主要介绍了jmeter添加自定函数的实例(jmeter5.3+IntelliJ IDEA),本文通过图文并茂的形式给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下2020-11-11
最新评论