Spring Boot 之遇见JSON
MVC框架中,Spring Boot内置了jackson来完成JSON的序列化和反序列化操作,并且,在与其他技术集成的时候,如Redis、MongoDB、Elasticsearch等对象序列化,都可使用jackson来完成。
Spring Boot 之遇见JSON
Spring Boot中使用Jackson
在Controller中,方法注解为@ResponseBody,则自动将方法返回的对象序列化成JSON。
@Controller
@RequestMapping("/json")
public class JsonController(){
@Autowired UserService userService;
@GetMapping("/user/{id}.json")
public @ResponseBody User showUserInfo(@PathVariable Long id){
User user = userService.getUserById(id);
return user;
}
}
自定义ObjectMapper
使用Java Config来自定义一个ObjectMapper,使用@Bean来配置,如:
@Configuration
public class JacksonConf{
@Bean
public ObjectMapper getObjectMapper() {
ObjectMapper objectMapper = new ObjectMapper();
objectMapper.setDateFormat(new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"));
return objectMapper;
}
}
获取当前时间:
@GetMapping("/now.json")
public @ResopnseBody Map datetime() {
Map map = new HashMap ();
map.put("time", new Date());
return map;
}
当访问now.json时,输出:
{"time": "2019-03-17 11:11:11"}
Jackson的三种使用方式
- 采用JsonParser来解析JSON,解析结果是一串Tokens,采用JsonGenerator来生成JSON,这是最底层的方式。
- 采用树遍历(Tree Traversing)方式,JSON被读入到JsonNode对象中,可以像操作XML DOM那样读取JSON。
- 采用DataBind方式,将POJO序列化成JSON,或者反序列化到POJO。
Jackson树遍历
树遍历通常适合没有POJO对应的JSON,如:
@Autowired ObjectMapper mapper;
@GetMapping("/readtree.json")
public @ResponseBody String readtree() throws JsonProcessingException,IOException{
String json = "{\"name\":\"aaa\",\"id\":10}";
JsonNode node = mapper.readTree(json);
String name = node.get("name").asText();
int id = node.get("id").asInt();
return "name:"+name+",id"+id;
}
readTree方法可以接受一个字符串或者字节数组、文件、InputStream等,返回JsonNode作为根节点。
JsonNode支持以下方法来读取JSON数据:
- asXXX,如asText、asBoolean、asInt等,读取JsonNode对应的值。
- isArray,用于判断JsonNode是否是数组,如果是数组,可以用get(i)来进行遍历,通过size()来获取长度。
- get(String),获取当前节点的子节点,返回JsonNode。
JSON规范要求Key是字符串,且用双引号
对象绑定
应用程序更多的是使用Java对象与JSON数据互相绑定,仅仅调用ObjectMapper的readValue来实现,如:
{"name":"aaa",age:10}
可以创建一个POJO对象来与JSON相对应:
public class User {
Long id;
String name;
// setter getter
}
然后使用readValue来反序列化上面的JSON字符串:
@GetMapping("/databind.json")
public @ResponseBody String databind() throws JsonProcessingException,IOException{
String json = "{\"name\":\"aaa\",\"id\":10}";
User user = mapper.readValue(json,User.class);
return "name:"+user.getName()+",id:"+user.getId();
}
将POJO序列化成JSON,使用mapper的writeValueAsString方法:
@GetMapping("/serialization.json")
public @ResponseBody String custom() throws JsonProcessingException {
User user = new User();
user.setId(11);
user.setName("hello");
String str = mapper.writeValueAsString(user);
return str;
}
mapper.writeValueAsString将对象序列化成JSON字符串,可以使用Jackson注解来对序列化的字段进行定制。
流式操作
树模型和数据绑定都是基于流式操作完成的,即通过JsonParser来解析JSON,形成JsonToken流,下面的代码是解析JSON:
@Autowired
ObjectMapper mapper;
@RequestMapping("/parser.html")
public @ResponseBody String parser() throws JsonParseException,IOException{
String json = "{\"name\":\"aaa\",\"id\":10}";
JsonFactory f = mapper.getFactory();
String key = null,value=null;
JsonParser parser = f.createParser(json);
JsonToken token = parser.nextToken();
if(token == JsonToken.FIELD_NAME){
key = parser.getCurrentName();
}
token = parser.nextToken();
value = parser.getValueAsString();
parser.close();
return key+","+value;
}
JsonParser的解析结果包含了一系列JsonToken,JsonToken是一个枚举类型,常用的START_OBJECT代表符号“{”;START_ARRAY和END_ARRAY代表"[“和”]",FIELD_NAME表示一个JSON Key;VALUE_STRING代表一个JSON Value,字符串类型;VALUE_NUMBER则表示一个整数类型。
判断Token的类型后,通过调用getValueAsXXX来获取其值,XXX是其值的类型。
@RequestMapping("/generator.html")
public @ResponseBody String generator() throws JsonParseException,IOException{
JsonFactory f = mapper.getFactory();
StringWriter sw = new StringWriter();
JsonGenerator g = f.createGenerator(sw);
g.writeStartObject();
g.writeStringField("name","aaa");
g.writeEndObject();
g.close();
return sw.toString();
}
Jackson注解
Jackson包含了很多注解,用来个性化序列化和反序列化,如:
@JsonProperty,作用在属性上,用来为JSON Key指定一个别名。
@JsonProperty("userName")
private String name
@JsonIgnore,作用在属性上,用来忽略此属性。
@JsonIgnore
private String pwd
@JsonIgnoreProperties,忽略一组属性,作用在类上;
@JsonIgnoreProperties
public static class SamplePojo{
}
@JsonAnySetter,标记在某个方法上,此方法接受Key、Value两个参数,用于JackSon在反序列化过程中,未找到的对应属性都调用此方法,通常用一个map来实现:
@JsonAnySetter
private void other( String property, Object value ) {
map.put(property,value);
}
@JsonAnyGetter,此注解标注在一个返回Map的方法上,JackSon会取出Map中的每一个值进行序列化。
Class Department {
Map map = new HashMap();
int id;
public Department(int id){
this.id = id;
map.put("newAttr",1);
}
@JsonAnyGetter
public Map<String,Object> getOtherProperties() {
return map;
}
}
Department,对象序列化的时候,其JSON类似:
{“id”:1,“newAttr”:1}
@JsonFormat,用于日期格式,
@JsonFormat(pattern = "yyyy-MM-dd HH-mm-ss")
private Date createDate;
@JsonNaming,用于指定一个命名策略,作用于类或者属性上。
@JsonSerialize,指定一个实现类来自定义序列化。类必须实现JsonSerializer接口。
@JsonDeserialize,用户自定义反序列化。
@JsonView,作用在类或者属性上,用来自定义一个序列化组。比如对于User对象,某些情况下只返回id,而某些情况下需要返回id和name;
public class User {
public interface IdView {};
public interface IdNameView extends IdView {};
@JsonView(IdView.class)
private Interger id;
@JsonView(IdNameView.class)
private String name;
}
集合的反序列化
在Spring MVC的Controller方法中,可以使用@RequestBody将提交的JSON自动映射到方法参数上,如:
@RequestMapping("/updateUsers.json")
public @ResponseBody String say(@RequestBody List<User> list){
StringBuilder sb = new StringBuilder();
for(User user:list){
sb.append(user.getName()).append(" ");
}
return sb.toString();
}
接受如下一个JSON请求,并自动映射到User对象上。
[
{
"name":"aaa",
"id":1
}
{
"name":"bbb",
"id":2
}
]
Spring Boot能自动识别出List对象包含的是User类,因为在方法中定义的泛型的类型会被保留在字节码中,所以Spring Boot能识别List包含的泛型类型从而能正确反序列化。
Spring Boot 之遇见JSON的更多相关文章
- spring boot @ResponseBody转换JSON 时 Date 类型处理方法,Jackson和FastJson两种方式,springboot 2.0.9配置fastjson不生效官方解决办法
spring boot @ResponseBody转换JSON 时 Date 类型处理方法 ,这里一共有两种不同解析方式(Jackson和FastJson两种方式,springboot我用的1.x的版 ...
- Spring Boot 之使用 Json 详解
Spring Boot 之使用 Json 详解 简介 Spring Boot 支持的 Json 库 Spring Web 中的序列化.反序列化 指定类的 Json 序列化.反序列化 @JsonTest ...
- Spring Boot Security And JSON Web Token
Spring Boot Security And JSON Web Token 说明 流程说明 何时生成和使用jwt,其实我们主要是token更有意义并携带一些信息 https://github.co ...
- Spring boot之返回json数据
1.步骤: 1. 编写实体类Demo 2. 编写getDemo()方法 3. 测试 2.项目构建 编写实体类Demo package com.kfit; /** * 这是一个测试实体类. */ pub ...
- Spring Boot HTTP over JSON 的错误码异常处理
摘要: 原创出处:www.bysocket.com 泥瓦匠BYSocket 希望转载,保留摘要,谢谢! “年轻人不要怕表现,要敢于出来表现,但还是那句话,要有正确的度,你的表现是分析问题和解决问题的能 ...
- Spring Boot默认的JSON解析框架设置
方案一:启动类继承WebMvcConfigurerAdapter,覆盖方法configureMessageConverters ... @SpringBootApplication public cl ...
- Spring boot中自定义Json参数解析器
转载请注明出处... 一.介绍 用过springMVC/spring boot的都清楚,在controller层接受参数,常用的都是两种接受方式,如下 /** * 请求路径 http://127.0. ...
- (4)Spring Boot使用别的json解析框架【从零开始学Spring Boot】
此文章已经废弃,请看新版的博客的完美解决方案: 78. Spring Boot完美使用FastJson解析JSON数据[从零开始学Spring Boot] http://412887952-qq-co ...
- spring boot 自己输出json数据
@RequestMapping("/json")public void json(HttpServletResponse response, Pager pager, TruckF ...
随机推荐
- webapplicationContext之ServletContext等相关概念说明
1)ServletContext是一个全局的储存信息的空间,所有用户共用一个,其信息必须是线程安全且共享的. ServletContext有一个接口定义:ServletContext接口.此接口定义了 ...
- AI算法测评事项
前言 注:大概2017年-2018年国内人工智能热度达到顶峰,随后热度开始逐渐减少.2018年前人工智能被投资界.学术界.工业界和媒体炒的特别热,各大企业都想尝试一下深度学习技术在业务场景的应用.试水 ...
- Java并发包源码学习系列:ReentrantLock可重入独占锁详解
目录 基本用法介绍 继承体系 构造方法 state状态表示 获取锁 void lock()方法 NonfairSync FairSync 公平与非公平策略的差异 void lockInterrupti ...
- hadoop目录结构
Hadoop目录结构 重要目录结构: bin目录:存放对Hadoop相关服务(HDFS,YARN)进行操作的脚本 etc目录:Hadoop的配置文件目录,存放Hadoop的配置文件 lib目录:存放H ...
- Dota游戏匹配的所有组合
在Dota游戏中有一种匹配玩法,任意5人以下玩家组队,加入匹配系统,由系统组合出5人 vs 5人的组合进行游戏,比如2人+3人 vs 1人+4人.抽象出这个问题,就变成两边各有m个玩家,最多允许n个 ...
- 一文彻底理解IO多路复用
在讲解IO多路复用之前,我们需要预习一下文件以及文件描述符. 什么是文件 程序员使用I/O最终都逃不过文件. 因为这篇同属于高性能.高并发系列,讲到高性能.高并发就离不开Linux/Unix,因此这里 ...
- Redis Cluster 集群节点维护 (三)
Redis Cluster 集群节点维护: 集群运行很久之后,难免由于硬件故障,网络规划,业务增长,等原因对已有集群进行相应的调整,比如增加redis nodes 节点,减少节点,节点迁移,更换服务器 ...
- 十八:SQL注入之堆叠及绕WAF
堆叠查询注入 (双查询注入) stacked injections(堆叠注入)从名词的含义就可以看到是一堆的SQL语句一起执行,而在真实的运用中也是这样的,我们知道在mysql中,主要是命令行中,每一 ...
- JavaScript入门-函数function(二)
JavaScript入门-函数function(二) 递归函数 什么是递归函数? 递归简单理解就是,在函数体里,调用自己. //我们在求一个10的阶乘的时候,可能会这么做 //写一个循环 var to ...
- RCE - Pikachu
概述: 远程系统命令执行 一般出现这种漏洞,是因为应用系统从设计上需要给用户提供指定的远程命令操作的接口 比如我们常见的路由器.防火墙.入侵检测等设备的web管理界面上 一般会给用户提供一个ping操 ...