当前位置:   article > 正文

Spring Boot 返回 Json 数据_springboot返回json数据

springboot返回json数据

1. Spring Boot 返回 Json 数据

XML 文件的解析常见的解析工具有 DOM4j、JDOM 等,为了标准化 XML 文件解析,Java 中提出了 JAXP 规范,使用的解析模型有

  1. DOM:将标记语言文档一次性加载进入内存中,在内存中形成一颗 DOM 树(服务器端常用)
    • 优点:操作方便,可以对文档进行 CRUD 的所有操作
    • 缺点:一次性加载进入内存形成 DOM 树,非常消耗资源
  2. SAX:逐行读取,基于事件驱动(安卓终端常用)
    • 优点:不消耗资源
    • 缺点:只能读取,不能增删改
    1. public class Test2 {
    2. public static void main(String[] args) throws Exception {
    3. SAXParserFactory parserFactory = SAXParserFactory.newInstance();
    4. SAXParser saxParser = parserFactory.newSAXParser();
    5. final List<Student> studentList = new ArrayList<Student>();
    6. saxParser.parse("xml02/students.xml", new DefaultHandler() {
    7. private Student student=null;
    8. private int flag=0;
    9. public void startElement(String uri, String localName, String qName, Attributes attributes) throws SAXException {
    10. if ("student".equals(qName)) {
    11. student=new Student();
    12. String id=attributes.getValue("id");//获取当前标签的指定名称的属性
    13. if(id!=null && id.trim().length()>0)
    14. student.setId(id);
    15. }else if("name".equals(qName))
    16. flag=1;
    17. else if("age".equals(qName))
    18. flag=2;
    19. else if("sex".equals(qName))
    20. flag=3;
    21. }
    22. @Override
    23. public void characters(char[] ch, int start, int length) throws SAXException {
    24. String tmp = new String(ch, start, length);
    25. if(tmp!=null && tmp.trim().length()>0){
    26. if(flag==1)
    27. student.setName(tmp.trim());
    28. else if(flag==2){
    29. Integer kk=Integer.parseInt(tmp.trim());
    30. student.setAge(kk);
    31. }else if(flag==3)
    32. student.setSex(tmp.trim());
    33. }
    34. }
    35. @Override
    36. public void endElement(String uri, String localName, String qName) throws SAXException {
    37. flag=0;
    38. if("student".equals(qName))
    39. studentList.add(student);
    40. }
    41. });
    42. studentList.forEach(System.out::println);
    43. }
    44. }

早期数据传输使用 xml 作为交互格式,例如 webservice 技术,但是由于 xml 解析比较麻烦,所以现在在项目开发中,在接口与接口之间以及前后端之间数据的传输都使用 Json 格式,在 Spring Boot 中接口返回 Json 格式的数据很简单,在 Controller 中使用@RestController 注解即可返回 Json 格式的数据,@RestController 也是 Spring Boot 新增的一个复合注解。

源代码:其中 Target、Retention 和 Documented 是元注解

@Target({ElementType.TYPE}) 用于声明注解可以使用在什么地方,Type 表示可以使用在类上

@Retention(RetentionPolicy.RUNTIME) 用于声明注解需要保持到什么阶段,Runtime 表示注解在编译生成的字节码中一直保持到运行时

@Documented

@Controller

@ResponseBody

public @interface RestController {
        String value() default "";
}

可以看出 @RestController 注解包含了原来的 @Controller 和 @ResponseBody 注解,使用过 Spring 对 @Controller 注解用于声明当前类是控制器类,@ResponseBody 注解是将返回的数据结构转换为 Json 格式。所以在默认情况下,使用了@RestController 注解即可将返回的数据结构转换成 Json 格式,Spring Boot 中默认使用的 Json 解析技术框架是 jackson。点开 pom.xml 中的 spring-boot-starter-web 依赖,可以看到一个 spring-boot-starter-json 依赖:

  1. <dependency>
  2. <groupId>org.springframework.boot</groupId>
  3. <artifactId>spring-boot-starter-json</artifactId>
  4. <scope>compile</scope>
  5. </dependency>

Spring Boot 中对依赖都做了很好的封装,可以看到很多 spring-boot-starter-xxx 系列的依赖,这是 Spring Boot 的特点之一,不需要人为去引入很多相关的依赖了,starter-xxx 系列直接都包含了所必要的依赖,所以再次点进去上面这个 spring-boot-starter-json 依赖,可以看到:

  1. <dependency>
  2. <groupId>com.fasterxml.jackson.core</groupId>
  3. <artifactId>jackson-databind</artifactId>
  4. <scope>compile</scope>
  5. </dependency>
  6. <dependency>
  7. <groupId>com.fasterxml.jackson.datatype</groupId>
  8. <artifactId>jackson-datatype-jdk8</artifactId>
  9. <scope>compile</scope>
  10. </dependency>
  11. <dependency>
  12. <groupId>com.fasterxml.jackson.datatype</groupId>
  13. <artifactId>jackson-datatype-jsr310</artifactId>
  14. <scope>compile</scope>
  15. </dependency>
  16. <dependency>
  17. <groupId>com.fasterxml.jackson.module</groupId>
  18. <artifactId>jackson-module-parameter-names</artifactId>
  19. <scope>compile</scope>
  20. </dependency>

到此为止知道了 Spring Boot 中默认使用的 json 解析框架是 jackson。默认的 jackson 框架对常用数据类型的转 Json 处理。

1. Spring Boot 默认对 Json 的处理

在实际项目中,常用的数据结构无非有类对象、List 对象、Map 对象,默认的 jackson 框架对这三个常用的数据结构都可以转成 json 的格式。

1.1 创建 User 实体类需要创建一个实体类 User。

  1. public class User {
  2.         private Long id;
  3.         private String username;
  4.         private String password;
  5.         /* 省略 get、set 和带参构造方法 */
  6. }

1.2 创建 Controller 类

然后创建一个 Controller,分别返回 User 对象、List<User>和 Map<String, Object>。

  1. import org.springframework.web.bind.annotation.RequestMapping;
  2. import org.springframework.web.bind.annotation.RestController;
  3. import java.util.ArrayList;
  4. import java.util.HashMap;
  5. import java.util.List;
  6. import java.util.Map;
  7. @RestController
  8. @RequestMapping("/json")
  9. public class JsonController {
  10. @RequestMapping("/user")
  11. public User getUser() {
  12. return new User(1, "葱花", "123456");
  13. }
  14. @RequestMapping("/list")
  15. public List<User> getUserList() {
  16. List<User> userList = new ArrayList<>();
  17. User user1 = new User(1, "葱花", "123456");
  18. User user2 = new User(2, "香菜", "123456");
  19. userList.add(user1);
  20. userList.add(user2);
  21. return userList;
  22. }
  23. @RequestMapping("/map")
  24. public Map<String, Object> getMap() {
  25. Map<String, Object> map = new HashMap<>(3);
  26. User user = new User(1, "葱花", "123456");
  27. map.put("作者信息", user);
  28. map.put("博客地址", "http://blog.yan.com");
  29. map.put("CSDN 地址", "http://blog.csdn.net/conghua");
  30. map.put("粉丝数量", 21);
  31. return map;
  32. }
  33. }

1.3 测试不同数据类型返回的 json

写好了接口,分别返回了一个 User 对象、一个 List 集合和一个 Map 集合,其中 Map 集合中的 value 存的是不同的数据类型

在浏览器中输入:localhost:8080/json/user 返回 json:

{"id":1,"username":"葱花","password":"123456"}

在浏览器中输入:localhost:8080/json/list 返回 json:

[{"id":1, "username":"葱花", "password":"123456"}, {"id":2, "username":"香菜", "password":"123456"}]

在浏览器中输入:`localhost:8080/json/map 返回 json:

{" 作 者 信 息 "       :   {"id":1,    "username":" 葱花 ",     "password":"123456"},                     "CSDN 地 址 "                      :

"http://blog.csdn.net/conghua", "粉丝数量":21, "博客地址":"http://blog.cong.com"}

可以看出 map 中不管是什么数据类型,都可以转成相应的 json 格式,这样就非常方便。

1.4 jackson 中对 null 的处理

在实际项目中,难免会遇到一些 null 值出现,转 json 时是不希望有这些 null 出现的,比如期望所有的 null 在转 json 时都变成""这种空字符串,那怎么做呢?在 Spring Boot 中做一下配置即可,新建一个 jackson 的配置类:

  1. import com.fasterxml.jackson.core.JsonGenerator;
  2. import com.fasterxml.jackson.databind.JsonSerializer;
  3. import com.fasterxml.jackson.databind.ObjectMapper;
  4. import com.fasterxml.jackson.databind.SerializerProvider;
  5. import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean;
  6. import org.springframework.context.annotation.Bean;
  7. import org.springframework.context.annotation.Configuration;
  8. import org.springframework.context.annotation.Primary;
  9. import org.springframework.http.converter.json.Jackson2ObjectMapperBuilder;
  10. import java.io.IOException;

@Configuration 用于声明当前类是一个配置类,最新的注解为@SpringBootConfiguration public class JacksonConfig {

@Bean 用于在配置类中,表示方法的返回值是一个受管 bean

@Primary 如果多个配置,则以当前的为主

        @ConditionalOnMissingBean(ObjectMapper.class) 条件注解,表示如果受管 bean 中没有 ObjectMapper 类型的对象,则需要构建

  1. public ObjectMapper jacksonObjectMapper(Jackson2ObjectMapperBuilder builder) {
  2. ObjectMapper objectMapper = builder.createXmlMapper(false).build();
  3. ObjectMapper.getSerializerProvider().setNullValueSerializer(new JsonSerializer<Object>() {
  4. @Override
  5. public void serialize(Object o, JsonGenerator jsonGenerator, SerializerProvider serializerProvider)
  6. throws IOException {
  7. jsonGenerator.writeString(""); 针对 null 内容输出空字符串
  8. }
  9. });
  10. return objectMapper;
  11. }

然后修改一下上面返回 map 的接口,将几个值改成 null 测试一下:

  1. @RequestMapping("/map")
  2. public Map<String, Object> getMap() {
  3. Map<String, Object> map = new HashMap<>(3);
  4. User user = new User(1, "葱花", null); map.put("作者信息", user);
  5. map.put("博客地址", "http://blog.cong.com");
  6. map.put("CSDN 地址", null);
  7. map.put("粉丝数量", 21);
  8. return map;
  9. }

重启项目再次输入 localhost:8080/json/map,可以看到 jackson 已经将所有 null 字段转成了空字符串了。例如返回数据为{"作者信息":{"id":1,"username":"葱花","password":""},"CSDN 地址":"","粉丝数量":21,"博客地

址":"http://blog.cong.com"}

配置属性不参与映射输出

@JsonIgnore 指定该属性不参与 JSON 字符串的映射

private String password;

配置日期类型数据的映射格式

@JsonFormat(pattern = "yyyy-MM-dd")

private Date birth=new Date();

2. 整合通用 Mapper 的开发方法

通用 mapper 就是基于 mybatis 的一款 MyBatis 增强插件,可以提供一些常用增、删、改、查的操作,不需要重复写一些常用的 sql。简化操作,精简代码,并且达到代码风格统一的目的。它的出现不是为了替代 mybatis,而是让 mybatis 的开发更方便。

添加依赖:

  1. <dependency>
  2. <groupId>tk.mybatis</groupId>
  3. <artifactId>mapper-spring-boot-starter</artifactId>
  4. <version>4.2.2</version>
  5. </dependency>

使用通用 mapper 的开发方式,采用注解定义映射关系,自动生成常见的 SQL 语句,不需要 xml 映射元文件

  1. /*
  2. 1.表名默认使用类名,驼峰转下划线(只对大写字母进行处理),如 TestUser 默认对应的表名为 test_user
  3. 2.表名可以使用@Table(name = "tableName")进行指定,对不符合第一条默认规则的可以通过这种方式指定表名.
  4. 3.字段默认和@Column 一样,都会作为表字段,表字段默认为 Java 对象的 Field 名字驼峰转下划线形式.
  5. 4.可以使用@Column(name = "fieldName")指定不符合第 3 条规则的字段名
  6. 5.使用@Transient 注解可以忽略字段,添加该注解的字段不会作为表字段使用.
  7. 6.建议一定是有一个@Id 注解作为主键的字段,可以有多个@Id 注解的字段作为联合主键.
  8. 7.默认情况下,实体类中如果不存在包含@Id 注解的字段,所有的字段都会作为主键字段进行使用(这种效率极低).
  9. 8.实体类可以继承使用,可以参考测试代码中的 tk.mybatis.mapper.model.UserLogin2 类.
  10. 9.由于基本类型,如 int 作为实体类字段时会有默认值 0,而且无法消除,所以实体类中建议不要使用基本类型.
  11. 10.@NameStyle 注解,用来配置对象名/字段和表名/字段之间的转换方式,该注解优先于全局配置 style,可选值:
  12. normal:使用实体类名/属性名作为表名/字段名
  13. camelhump:这是默认值,驼峰转换为下划线形式
  14. uppercase:转换为大写
  15. lowercase:转换为小写
  16. */

@Entity    //用于声明当前类是一个实体类

@Table(name="tbl_users")    // 用于声明当前类所对应的表名称

public class User implements Serializable {

@Id //用于声明标识属性,对应表中的主键

@GeneratedValue(strategy = GenerationType.IDENTITY) 声明主键生成策略

private Long id;

@Column     //如果属性名称和列名称不一致,则需要通过@Column 进行配置对应的列名称 private String username;

private String password;

private Date birth;

private Boolean sex;

}

定义 mapper 接口

public interface UserMapper extends BaseMapper<User> {

}

针对 mapper 接口进行注册,一般是依赖自动扫描实现。可以在主类或者配置上添加一个注解配置

SpringBoot 针对 Junit 的单元测试有很好的支持

@SpringBootTest

class CommonMapperApplicationTests {

@Autowired

private UserMapper userMapper;

@Test

void contextLoads() {

}

@Test

void testCreate(){ User user=new User(); user.setUsername("张三丰"); user.setPassword("333333");

user.setSex(true); int len = userMapper.insertSelective(user); Assertions.assertEquals(1,len);

}

}

需要控制器调用业务,业务再通过 Mapper 访问数据库,最终返回 JSON 字符串

3. 使用阿里巴巴 FastJson 的设置

1. jackson 和 fastJson 的对比

有很多人已经习惯于使用阿里巴巴的 fastJson 来做项目中 json 转换的相关工作,比如目前项目中使用的就是

阿里的 fastJson

选项

fastJson

Jackson

上手难易程度

容易

中等

高级特性支持

中等

丰富

官方文档、Example 支持

中文

英文

处理 json 速度

略快

关于 fastJson 和 jackson 的对比,网上有很多资料可以查看,主要是根据自己实际项目情况来选择合适的框架。从扩展上来看,fastJson 没有 jackson 灵活,从速度或者上手难度来看,fastJson 可以考虑,项目中目前使用的是阿里的 fastJson,挺方便的。

2. fastJson 依赖导入

使用 fastJson 需要导入依赖

<dependency>

<groupId>com.alibaba</groupId>

<artifactId>fastjson</artifactId>

</dependency>

3. 使用 fastJson 处理 null

使用 fastJson 时,对 null 的处理和 jackson 有些不同,需要继承 WebMvcConfigurationSupport 类或者实现 WebMvcConfiguration 接口,然后覆盖 configureMessageConverters 方法,在方法中可以选择对要实现 null 转换的场景,配置好即可。

  1. import com.alibaba.fastjson.serializer.SerializerFeature; import com.alibaba.fastjson.support.config.FastJsonConfig; import com.alibaba.fastjson.support.spring.FastJsonHttpMessageConverter; import org.springframework.context.annotation.Configuration; import org.springframework.http.MediaType;
  2. import org.springframework.http.converter.HttpMessageConverter;
  3. import org.springframework.web.servlet.config.annotation.WebMvcConfigurationSupport; import java.nio.charset.Charset; import java.util.ArrayList; import java.util.List;
  4. @Configuration
  5. public class fastJsonConfig extends WebMvcConfigurationSupport {
  6. //使用阿里 FastJson 作为 JSON MessageConverter
  7. @Override
  8. public void configureMessageConverters(List<HttpMessageConverter<?>> converters) {
  9. FastJsonHttpMessageConverter converter = new FastJsonHttpMessageConverter();
  10. FastJsonConfig config = new FastJsonConfig(); config.setSerializerFeatures(
  11. SerializerFeature.WriteMapNullValue, // 保留 map 空的字段
  12. SerializerFeature.WriteNullStringAsEmpty, // 将 String 类型的 null 转成""
  13. SerializerFeature.WriteNullNumberAsZero, // 将 Number 类型的 null 转成 0
  14. SerializerFeature.WriteNullListAsEmpty, // 将 List 类型的 null 转成[]
  15. SerializerFeature.WriteNullBooleanAsFalse, // 将 Boolean 类型的 null 转成 false
  16. SerializerFeature.DisableCircularReferenceDetect); // 避免循环引用
  17. converter.setFastJsonConfig(config);
  18. converter.setDefaultCharset(Charset.forName("UTF-8"));
  19. List<MediaType> mediaTypeList = new ArrayList<>();
  20. // 解决中文乱码问题,相当于在 Controller 上的@RequestMapping 中加了个属性 produces = "application/json" mediaTypeList.add(MediaType.APPLICATION_JSON);
  21. converter.setSupportedMediaTypes(mediaTypeList); converters.add(converter);
  22. }
  23. }

4. 测试工具 postman

如果直接测试,则需要编写页面和 js 代码才能进行验证,可以使用 postman 避免这些繁琐的操作

 

这里针对日期类型数据的格式转换会有报错。最终提交数据为

{"username":"谢逊","sex":"true","birth":"1989-02-03","password":"666666"}

4. 封装统一返回的数据结构

在实际项目中,除了要封装数据之外,往往需要在返回的 json 中添加一些其他信息,比如返回一些状态码 code 【注意不是 response 的响应状态码】,返回一些 msg 给调用者,这样调用者可以根据 code 或者 msg 做一些逻辑判断。所以在实际项目中,需要封装一个统一的 json 返回结构存储返回信息。

4.1 定义统一的 json 结构

由于封装的 json 数据的类型不确定,所以在定义统一的 json 结构时需要用到泛型。统一的 json 结构中属性包括数据、状态码、提示信息即可,构造方法可以根据实际业务需求做相应的添加即可,一般来说,应该有默认的返回结构,也应该有用户指定的返回结构。

public class JsonResult<T> {

         private T data;   //需要传递的数据

private int code; //用户自定义相应码,注意不是服务器响应状态码。如果不需要传递详细信息还可以使用 boolean success;

         private String msg;   // 服务器回传信息

public JsonResult() { 若没有数据返回,默认状态码为 0,提示信息为:操作成功! this.code = 0;

this.msg = "操作成功!";

}

public JsonResult(int code, String msg) { 若没有数据返回,可以人为指定状态码和提示信息 this.code = code; this.msg = msg;

}

public JsonResult(T data) { 有数据返回时,状态码为 0,默认提示信息为:操作成功! this.data = data; this.code = 0;

this.msg = "操作成功!";

}

public JsonResult(T data, String msg) { 有数据返回,状态码为 0,人为指定提示信息 this.data = data; this.code = 0; this.msg = msg;

}

// 省略 get 和 set 方法

}

建议写法,不一定最佳

4.2 修改 Controller 中的返回值类型及测试

由于 JsonResult 使用了泛型,所以所有的返回值类型都可以使用该统一结构,在具体的场景将泛型替换成具体的数据类型即可,非常方便,也便于维护。在实际项目中,还可以继续封装,比如状态码和提示信息可以定义一个枚举类型,以后只需要维护这个枚举类型中的数据即可。根据以上的 JsonResult 可以改写一下 Controller

  1. @RestController
  2. @RequestMapping("/jsonresult") public class JsonResultController { @RequestMapping("/user") public JsonResult<User> getUser() {
  3. User user = new User(1, "葱花", "123456");
  4. return new JsonResult<>(user);
  5. }
  6. @RequestMapping("/list")
  7. public JsonResult<List> getUserList() {
  8. List<User> userList = new ArrayList<>();
  9. User user1 = new User(1, "", "123456");
  10. User user2 = new User(2, "大葱花", "123456");
  11. userList.add(user1); userList.add(user2);
  12. return new JsonResult<>(userList, "获取用户列表成功");
  13. }
  14. @RequestMapping("/map") public JsonResult<Map> getMap() {
  15. Map<String, Object> map = new HashMap<>(3); User user = new User(1, "葱花", null); map.put("作者信息", user);
  16. map.put("博客地址", "http://blog.yan.com"); map.put("CSDN 地址", null); map.put("粉丝数量", 4153); return new JsonResult<>(map); }
  17. }

重新在浏览器中输入:localhost:8080/jsonresult/user 返回 json:

{"code":0,"data":{"id":1,"password":"123456","username":"葱花"},"msg":"操作成功!"} 输入:localhost:8080/jsonresult/list,返回 json 格式:

{"code":0,  "data":[{"id":1,  "password":"123456",  "username":" 葱花"},   {"id":2,            "password":"123456",

"username":"香菜"}], "msg":"获取用户列表成功"} 输入:localhost:8080/jsonresult/map,返回 json 格式:

{"code":"0", "data":{"作者信息":{"id":1, "password":"","username":"葱花"},"CSDN 地址":null,"粉丝数量

":21,"博客地址":"http://blog.cong.com"},"msg":"操作成功!"}

通过封装,不但将数据通过 json 传给前端或者其他接口,还带上了状态码和提示信息,这在实际项目场景中应用非常广泛。

5. 总结

这里主要是对 Spring Boot 中 json 数据的返回做了详细的分析,从 Spring Boot 默认的 jackson 框架到阿里的 fastJson 框架,分别对它们的配置做了相应的描述。另外,结合实际项目情况,总结了实际项目中使用的 json 封装结构体,加入了状态码和提示信息,使得返回的 json 数据信息更加完整。

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/我家自动化/article/detail/619855
推荐阅读
相关标签
  

闽ICP备14008679号