SpringBoot笔记--Jackson
SpringUtil.getBean<GenericConversionService>().addConverter(Date2LocalDateTimeConverter())
var handerAdapter = SpringUtil.context.getBean(RequestMappingHandlerAdapter::class.java);
var listResolvers = mutableListOf<HandlerMethodArgumentResolver>()
listResolvers.add(RequestParameterConverter(listOf(ApiParam::class.java)));
listResolvers.addAll(handerAdapter.argumentResolvers)
handerAdapter.argumentResolvers = listResolvers;
internal class MyJsonMapper
@JvmOverloads private constructor()
: ObjectMapper() {
init {
// 设置输出时包含属性的风格
this.findAndRegisterModules();
this.setSerializationInclusion(JsonInclude.Include.NON_EMPTY)
// 允许单引号、允许不带引号的字段名称
this.configure(JsonParser.Feature.ALLOW_SINGLE_QUOTES, true)
this.configure(JsonParser.Feature.ALLOW_UNQUOTED_FIELD_NAMES, true)
// 设置输入时忽略在JSON字符串中存在但Java对象实际没有的属性
this.disable(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES)
// 空值处理为空串
this.serializerProvider.setNullValueSerializer(object : JsonSerializer<Any>() {
@Throws(IOException::class, JsonProcessingException::class)
override fun serialize(value: Any, jgen: JsonGenerator,
provider: SerializerProvider) {
jgen.writeString("")
}
})
this.registerModule(CustomModule());
// 进行HTML解码。
// this.registerModule(SimpleModule().addSerializer(String::class.java, object : JsonSerializer<String>() {
// @Throws(IOException::class, JsonProcessingException::class)
// override fun serialize(value: String, jgen: JsonGenerator,
// provider: SerializerProvider) {
// jgen.writeString(StringEscapeUtils.unescapeHtml4(value))
// }
// }))
// 设置时区
this.setTimeZone(TimeZone.getDefault())//getTimeZone("GMT+8:00")
}
companion object {
/**
* 创建只输出非Null且非Empty(如List.isEmpty)的属性到Json字符串的Mapper,建议在外部接口中使用.
*/
val instance: MyJsonMapper = MyJsonMapper();
}
}
// 先定义一个json生成器
class ObjectIdJsonSerializer : JsonSerializer<ObjectId>() {
override fun serialize(o: ObjectId?, j: JsonGenerator, s: SerializerProvider) {
if (o == null) {
j.writeNull()
} else {
j.writeString(o.toString())
}
}
}
class LocalDateJsonSerializer : JsonSerializer<LocalDate>() {
override fun serialize(value: LocalDate?, generator: JsonGenerator, serializers: SerializerProvider) {
if (value == null) {
generator.writeNull()
} else {
generator.writeNumber(Date.from(value.atStartOfDay().atZone(ZoneId.systemDefault()).toInstant()).time)
}
}
}
class LocalTimeJsonSerializer : JsonSerializer<LocalTime>() {
override fun serialize(value: LocalTime?, generator: JsonGenerator, serializers: SerializerProvider) {
if (value == null) {
generator.writeNull()
} else {
generator.writeString(value.AsString())
}
}
}
class LocalDateTimeJsonSerializer : JsonSerializer<LocalDateTime>() {
override fun serialize(value: LocalDateTime?, generator: JsonGenerator, serializers: SerializerProvider) {
if (value == null) {
generator.writeNull()
} else {
generator.writeNumber(Date.from(value.atZone(ZoneId.systemDefault()).toInstant()).time)
}
}
}
class LocalDateJsonDeserializer : JsonDeserializer<LocalDate>() {
override fun deserialize(json: JsonParser?, ctxt: DeserializationContext?): LocalDate? {
if (json == null) {
return null;
}
if (json.valueAsString.contains("-")) {
return json.valueAsString.AsLocalDateTime().toLocalDate();
}
return Date(json.longValue).AsLocalDateTime().toLocalDate();
}
}
class LocalTimeJsonDeserializer : JsonDeserializer<LocalTime>() {
override fun deserialize(json: JsonParser?, ctxt: DeserializationContext?): LocalTime? {
if (json == null) {
return null;
}
if (json.valueAsString.contains(".")) {
return LocalTime.parse(json.valueAsString, DateTimeFormatter.ofPattern("HH:mm:ss.sss"))
}
return LocalTime.parse(json.valueAsString, DateTimeFormatter.ofPattern("HH:mm:ss"))
}
}
class LocalDateTimeJsonDeserializer : JsonDeserializer<LocalDateTime>() {
override fun deserialize(json: JsonParser?, ctxt: DeserializationContext?): LocalDateTime? {
if (json == null) {
return null;
}
if (json.valueAsString.contains("-")) {
return json.valueAsString.AsLocalDateTime();
}
return Date(json.longValue).AsLocalDateTime();
}
}
//Jackson 输出的时候,进行自定义序列化格式。
// http://www.jianshu.com/p/a0fb6559f56d
@Component
class CustomModule() : SimpleModule(PackageVersion.VERSION) {
init {
addSerializer(ObjectId::class.java, ObjectIdJsonSerializer());
addSerializer(LocalDate::class.java, LocalDateJsonSerializer());
addSerializer(LocalTime::class.java, LocalTimeJsonSerializer());
addSerializer(LocalDateTime::class.java, LocalDateTimeJsonSerializer());
addDeserializer(LocalDate::class.java, LocalDateJsonDeserializer())
addDeserializer(LocalTime::class.java, LocalTimeJsonDeserializer())
addDeserializer(LocalDateTime::class.java, LocalDateTimeJsonDeserializer())
}
}
fun <T> T.ToJson(): String {
if (this is String) return this;
return MyJsonMapper.instance.writeValueAsString(this) ?: ""
}
fun <T> String.FromJson(collectionClass: Class<T>): T {
if (collectionClass == String::class.java) {
return this as T
}
var jsonString = this.RemoveComment().Remove("\r\n", "\n")
if (jsonString.isEmpty()) {
return collectionClass.newInstance();
}
return MyJsonMapper.instance.readValue(jsonString, collectionClass)!!
}
inline fun <reified T> String.FromJson(): T {
return this.FromJson(T::class.java)
}
class RequestParameterConverter(var canIgnoreAnnotationClazzes: List<Class<*>> = listOf()) : HandlerMethodArgumentResolver {
override fun supportsParameter(parameter: MethodParameter?): Boolean {
if (parameter == null) return false
if (ServletRequest::class.java.isAssignableFrom(parameter.parameterType)) return false
if (ServletResponse::class.java.isAssignableFrom(parameter.parameterType)) return false
if (HttpSession::class.java.isAssignableFrom(parameter.parameterType)) return false
if (parameter.hasParameterAnnotation(PathVariable::class.java)) return false;
if (parameter.hasParameterAnnotation(CookieValue::class.java)) return false;
if (parameter.hasParameterAnnotation(RequestHeader::class.java)) return false;
if (parameter.hasParameterAnnotation(RequestParam::class.java)) return false;
var parameterAnnotations = parameter.parameterAnnotations.toMutableList();
parameterAnnotations.removeAll { Nullable::class.java.isInstance(it) }
parameterAnnotations.removeAll { NotNull::class.java.isInstance(it) }
canIgnoreAnnotationClazzes.forEach { ignoreClazz ->
parameterAnnotations.removeAll { ignoreClazz.isInstance(it) }
}
if (parameterAnnotations.size > 0) return false;
return true
//return parameter?.hasParameterAnnotation(JsonValue::class.java) ?: false;
}
override fun resolveArgument(parameter: MethodParameter?, mavContainer: ModelAndViewContainer?, webRequest: NativeWebRequest?, binderFactory: WebDataBinderFactory?): Any? {
if (parameter == null || mavContainer == null || webRequest == null || binderFactory == null) return null
var key = parameter.parameterName
var request = (webRequest as ServletWebRequest).request as MyHttpRequestWrapper;
var value: Any? = null
if (request.json.containsKey(key)) {
value = request.json.get(key)
} else if (request.requestMap.contains(key)) {
value = request.requestMap.get(request.queryString);
}
if (value == null || (value is String && value.isEmpty())) {
if (parameter.parameterType == String::class.java) {
return "";
} else if (Number::class.java.isAssignableFrom(parameter.parameterType)) {
return MyUtil.getSimpleClassDefaultValue(parameter.parameterType);
} else if (parameter.parameterType.isArray) {
return "[]".FromJson(parameter.parameterType)
} else if (Iterable::class.java.isAssignableFrom(parameter.parameterType)) {
return "[]".FromJson(parameter.parameterType);
}
return null;
// var nullable = parameter.parameterAnnotations.any { Nullable::class.java.isInstance(it) }
//
// if (nullable) {
// return null;
// }
// return MyUtil.getSimpleClassDefaultValue(parameter.parameterType)
}
if (parameter.parameterType.isEnum) {
return value.AsString().ToEnum(parameter.parameterType)
} else if (value.javaClass.isInstance(parameter.parameterType)) {
return value;
} else if (parameter.parameterType == Boolean::class.java) {
return value.AsBoolean()
} else if (parameter.parameterType == Character::class.java) {
return value.toString()[0]
} else if (parameter.parameterType == Byte::class.java) {
return value.AsInt().toByte()
} else if (parameter.parameterType == Short::class.java) {
return value.AsInt().toShort()
} else if (parameter.parameterType == Int::class.java ||
parameter.parameterType == java.lang.Integer::class.java) {
return value.AsInt()
} else if (parameter.parameterType == Long::class.java) {
return value.AsLong()
} else if (parameter.parameterType == Float::class.java) {
return value.AsDouble()
} else if (parameter.parameterType == Double::class.java) {
return value.AsDouble().toLong()
} else if (parameter.parameterType == String::class.java) {
return value.AsString()
} else if (parameter.parameterType == LocalDate::class.java) {
return value.AsLocalDateTime().toLocalDate()
} else if (parameter.parameterType == LocalTime::class.java) {
return value.AsLocalDateTime().toLocalTime()
} else if (parameter.parameterType == LocalDateTime::class.java) {
return value.AsLocalDateTime()
} else if (parameter.parameterType == Date::class.java) {
return Date.from(value.AsLocalDateTime().atZone(ZoneId.systemDefault()).toInstant());
} else {
return value.AsString().FromJson(parameter.parameterType);
}
return value
}
}
@Configuration
open class MongoConfig {
@Bean
@Throws(Exception::class)
open fun mongoTemplate(): MongoTemplate {
var dbFactory = SpringUtil.getBean<MongoDbFactory>();
//remove _class
val converter = MappingMongoConverter(DefaultDbRefResolver(dbFactory), MongoMappingContext())
converter.typeMapper = DefaultMongoTypeMapper(null)
(converter.conversionService as GenericConversionService).addConverter(Date2LocalDateTimeConverter())
return MongoTemplate(dbFactory, converter)
}
}
class Date2LocalDateTimeConverter : GenericConverter {
override fun getConvertibleTypes(): MutableSet<GenericConverter.ConvertiblePair> {
var pairs = hashSetOf<GenericConverter.ConvertiblePair>();
pairs.add(GenericConverter.ConvertiblePair(String::class.java, LocalDate::class.java));
pairs.add(GenericConverter.ConvertiblePair(String::class.java, LocalTime::class.java));
pairs.add(GenericConverter.ConvertiblePair(String::class.java, LocalDateTime::class.java));
pairs.add(GenericConverter.ConvertiblePair(Date::class.java, LocalDate::class.java));
pairs.add(GenericConverter.ConvertiblePair(Date::class.java, LocalTime::class.java));
pairs.add(GenericConverter.ConvertiblePair(Date::class.java, LocalDateTime::class.java));
pairs.add(GenericConverter.ConvertiblePair(LocalDate::class.java, Date::class.java));
pairs.add(GenericConverter.ConvertiblePair(LocalTime::class.java, Date::class.java));
pairs.add(GenericConverter.ConvertiblePair(LocalDateTime::class.java, Date::class.java));
return pairs;
}
override fun convert(value: Any?, sourceType: TypeDescriptor?, targetType: TypeDescriptor?): Any? {
if (value == null || sourceType == null || targetType == null) return null
var valueClass = sourceType.type;
var targetClass = targetType.type;
if (valueClass == String::class.java) {
var strValue = value.AsString();
if (strValue.isEmpty()) {
return null;
}
if (targetClass == LocalDate::class.java) {
return strValue.AsLocalDateTime().toLocalDate();
}
if (targetClass == LocalTime::class.java) {
return strValue.AsLocalDateTime().toLocalTime()
}
if (targetClass == LocalDateTime::class.java) {
return strValue.AsLocalDateTime();
}
}
if (valueClass == Date::class.java) {
if (targetClass == LocalDate::class.java) {
return value.AsLocalDateTime().toLocalDate();
}
if (targetClass == LocalTime::class.java) {
return value.AsLocalDateTime().toLocalTime();
}
if (targetClass == LocalDateTime::class.java) {
return value.AsLocalDateTime()
}
}
if (targetClass == Date::class.java) {
if (valueClass == LocalDate::class.java) {
return Date.from((value as LocalDate).atStartOfDay().atZone(ZoneId.systemDefault()).toInstant());
}
if (valueClass == LocalTime::class.java) {
return Date.from((value as LocalTime).atDate("1970-01-01".AsLocalDateTime().toLocalDate()).atZone(ZoneId.systemDefault()).toInstant());
}
if (valueClass == LocalDateTime::class.java) {
return Date.from((value as LocalDateTime).atZone(ZoneId.systemDefault()).toInstant());
}
}
return null;
}
}
问题
返回实体时,如:
- JsonTypedResult 类型, 当有消息时, 仅返回: JsonResult
- JsonTypedResult 类型, 当 data.id 为 "000000000" 时, 返回 id= ""
SpringBoot笔记--Jackson的更多相关文章
- springboot之jackson的两种配置方式
springboot 针对jackson是自动化配置的,如果需要修改,有两种方式: 方式一:通过application.yml 配置属性说明:## spring.jackson.date-format ...
- SpringBoot与jackson.databind兼容报错问题
SpringBoot与jackson.databind兼容报错问题 ———————————————— 1.SpringBoot版本V2.0.0其依赖的jackson-databind版本为V2.9.4 ...
- 源码分析springboot自定义jackson序列化,默认null值个性化处理返回值
最近项目要实现一种需求,对于后端返回给前端的json格式的一种规范,不允许缺少字段和字段值都为null,所以琢磨了一下如何进行将springboot的Jackson序列化自定义一下,先看看如何实现,再 ...
- spring学习笔记---Jackson的使用和定制
前言: JAVA总是把实体对象(数据库/Nosql等)转换为POJO对象再处理, 虽然有各类框架予以强力支持. 但实体对象和POJO, 由于"饮食习惯", "民族特色 ...
- SpringBoot笔记一
1 开始 1.1 spring介绍 Spring Boot使开发独立的,产品级别的基于Spring的应用变得非常简单,你只需"just run". 我们为Spring平台及第三方库 ...
- SpringBoot笔记十六:ElasticSearch
目录 ElasticSearch官方文档 ElasticSearch安装 ElasticSearch简介 ElasticSearch操作数据,RESTful风格 存储 检查是否存在 删除 查询 更新 ...
- 让SpringBoot的jackson支持JavaBean嵌套的protobuf
问题背景 REST 项目使用protobuf 来加速项目开发,定义了很多model,vo,最终返回的仍然是JSON. 项目中一般使用 一个Response类, public class Respons ...
- SpringBoot系列——Jackson序列化
前言 Spring Boot提供了与三个JSON映射库的集成: Gson Jackson JSON-B Jackson是首选的默认库. 官网介绍: https://docs.spring.io/spr ...
- SpringBoot全局Jackson配置未生效
在做一个小项目,后台服务第一次用SpringBoot构建.接口使用Json格式,在application.properties中配置如下: spring.jackson.default-propert ...
随机推荐
- 使用wxpy自动发送微信消息(加强版)
通过使用wxpy自动发送微信消息后,笔者又加强了发送消息,堪称消息爆炸式发送 目前设置的为10秒发送一次,发送9次,每次发送10条内容 import requests import wxpy from ...
- tkinter做一个简单的登陆页面(十六)
做一个简单的登陆页面 import tkinter wuya = tkinter.Tk() wuya.title("wuya") wuya.geometry("900x3 ...
- 洗礼灵魂,修炼python(13)--模块random,math,pickle
random 1.作用: random模块用于生成随机数 2.常用函数: random:用于生成一个0到1的随机符点数: 0 <= n < 1.0 uniform(a, b):用于生成一个 ...
- MySQL5.7中的sql_mode默认值
简介 在正常项目开发过程中,如果MySQL版本从5.6升级到5.7版本.作为DBA在考虑数据库版本升级带来的影响时,一般会有几个注意点: sql_mode 默认值的改变 optimizer_switc ...
- exec 动态脚本 里面的参数和sp_executesql (注意引号,否则容易异常)
@indexCt int@DemographicName nvarchar(500)INSERT INTO #finalTemp EXEC('SELECT a.QuestionId,a.AnswerI ...
- IntelliJ IDEA 项目结构旁边出现 0%classes,0% lines covered
不知道一不小心点到哪里,项目变成如下形式 使用ctrl + Alt + F6弹出如下框,取消勾选-->点击Show Selected就可以去掉了 官网解释
- Git的上传步骤
Git的上传步骤 1.Git的命令基础 Git是当下最流行的版本控制工具(VCS),由linux系统之父linus开发.它能实现 团队中的代码协作开发,它在代码同步和代码管理方面功能强大,理念先进. ...
- 5分钟入门Tornado
Tornado 是 FriendFeed 使用的可扩展的非阻塞式 web 服务器及其相关工具的开源版本.这个 Web 框架看起来有些像web.py 或者 Google 的 webapp,不过为了能有效 ...
- hashCode相关性能优化
学习下hashMap中用到的关于hashCode性能优化技巧.作为笔记.为之后并发深入作基础. 1.关于提高性能的hash算法 在被模的位数为2的n次方时,用位与取代效率低下的模运算.位与效率相比模运 ...
- Arduino IDE for ESP8266 项目云盒子(2)一键自配置+网页服务器
https://item.taobao.com/item.htm?spm=a230r.1.14.20.eYblO3&id=521945102409&ns=1&abbucket= ...