前言

在前后端交互的选择上,之前一直采用的是模板引擎(因为我只负责后端)。
而这次的一个算是作业吧,前后端都是我,所以就研究了一下JSON交互在java web的应用(主要是前端)。

优缺点

  • 前后端耦合

    • 模板引擎加载只是将jsp的交互方式移植到html上,前端文件格式改变了,但是jsp中前后端耦合的缺点没有改变。
    • json交互中,数据通过js/jquery动态加载在页面上,数据与页面进行分离,页面只是单纯用于展示。
  • 数据加载逻辑复用

    • 模板引擎的方式中,如果有很多相似的页面元素以及一样的数据返回格式,那只是复制粘贴大法了。
    • 在模板引擎的例子中,只需要定义一套数据加载模型,传入不同的页面元素id以及数据则能实现逻辑复用
  • 后端接口的复用
    因为我的学习路线的问题,所以我开发过安卓原生一段时间。

    • 模板引擎式的加载必须使用webView组件加载,且需另进行原生构建时接口也要另外构建。
    • json交互,谷歌爸爸鼓励使用json进行交互(一年多前的事,现在不清楚了),且安卓原生内置GJSON进行json解析与构建,所以可以在原生以及跨平台的构建有很好的平衡。

对比

既然选择了json交互的方式,而java官方据我所知是没有内置对json的支持,Spring在Controller的层面使用RestContrller注解实现对json的支持。
但是我个人强迫症很强,我对Contrller(或者说是Presenter)的定义是

  • 对用户访问的url作页面的映射
  • 对用户触发的事件进行数据的传递与返回
    正是这两点的定义,我需要在Service层组合统一格式的结果返回到上层,所以需要第三方json支持。
    可选择的JSON库有很多,GSON,FastJson,Jackson,根据对比,我选择为马老师充值一波。

需求拆分

我初步定义的需求主要有三个

  • Service传递统一处理结果到上层
  • 无论是单一数据实体或者List型数据处理的结果是一样的
  • 处理过程是独立的,不依赖于实体类的支持

格式设计

在网上看过很多后端返回数据的格式,很多都是返回一个处理的status以及具体的数据,而这个status是根据http状态码进行设定的,因为这次时间比较紧,所以我就采用了这个方案。

  1. {
  2. "status": "status",
  3. "object": {
  4. }
  5. }

代码设计

因为我希望Controller能直接拿到结果,所以构建结果的过程全放在Result类中。
而结果构建我主要分为两种:只有状态码(通知处理结果)以及具有返回结果(数据显示),而根据结果的个人也分为两种:单个数据以及List型数据。

实际代码

  1. @Getter
  2. @Slf4j
  3. public class ResultSet {
  4. private JSONObject result;
  5. public ResultSet(){
  6. result=new JSONObject();
  7. }
  8. /**
  9. * 初始化状态码
  10. * @param status
  11. */
  12. public void initStatus(String status){
  13. result.put("status", status);
  14. }
  15. /**
  16. * 初始化状态码以及返回数据
  17. * @param status
  18. * @param obj
  19. */
  20. public void initData(String status,Object obj){
  21. initStatus(status);
  22. if(obj instanceof List){
  23. List list=(List)obj;
  24. JSONArray array=new JSONArray();
  25. for(Object object:list){
  26. array.add(putObjectToJSON(object));
  27. }
  28. result.put("object",array);
  29. }
  30. else {
  31. result.put("object",putObjectToJSON(obj));
  32. }
  33. }
  34. /**
  35. * 将单个Object放入json文件中
  36. * @param obj
  37. * @return
  38. */
  39. private JSONObject putObjectToJSON(Object obj){
  40. JSONObject result=new JSONObject();
  41. Field[] fields=obj.getClass().getDeclaredFields();
  42. for(Field field:fields){
  43. field.setAccessible(true);
  44. String fieldName=field.getType().getSimpleName();
  45. if(fieldName.equals("Department")||fieldName.equals("Job")){
  46. JSONObject tempJson=new JSONObject();
  47. try {
  48. Object tempObject = field.get(obj);
  49. Field[] tempFields=tempObject.getClass().getDeclaredFields();
  50. for(Field tempField:tempFields){
  51. tempField.setAccessible(true);
  52. tempJson.put(tempField.getName(),tempField.get(tempObject));
  53. tempField.setAccessible(false);
  54. }
  55. } catch (IllegalAccessException e) {
  56. e.printStackTrace();
  57. }
  58. if(fieldName.equals("Department")){
  59. result.put("depart",tempJson);
  60. }
  61. else {
  62. result.put("job",tempJson);
  63. }
  64. }
  65. else {
  66. try {
  67. result.put(field.getName(),field.get(obj));
  68. } catch (IllegalAccessException e) {
  69. e.printStackTrace();
  70. }
  71. }
  72. field.setAccessible(false);
  73. }
  74. return result;
  75. }
  76. }

代码思路

  1. Service通过initStatus/initData传入数据/状态码进行对象的生成

  2. Controller通过result的getter方法获取处理结果

  3. 单个数据以及List数据的处理

    1. 单个对象直接通过putObjectToJSON进行处理
    2. List数据通过对象类型判断,向下转型,遍历元素形成JSONArray进行处理,对元素处理的方法也是采用putObjectToJSON
  4. 嵌套对象的处理

    1. public class Employee {
    2. private int id;
    3. private Department depart;
    4. private Job job;
    5. private String name;
    6. ..................
    7. }
    1. 通过反射获取对象的所有成员变量类型以及对应的值
    2. 遇到上述bean类含有嵌套自定义对象时,递归生成json文件加入到结果json中

结果演示

​ (左为单object型,右为List型)

不足之处

  1. 状态码的设置应该采用枚举类的赋值,能更好的约束返回的状态码
  2. 在对象转换方面,应该采用配置扫描的方式。在配置文件中写入bean的包所在,在嵌套对象转换时通过扫描配置文件的信息判断
  3. 在嵌套对象的转换方面,只是做了一层的嵌套转换,更多层的没有考虑到,之后会再重构的
  4. 对异常处理方面,只是简单的输出错误信息。应该对错误信息进行进一步的处理

后记

因为这次时间比较赶,从项目的立项到成品的建立花了5天时间,所以注意到很多细节,但是没有去处理。
有想过之后有时间的话将这个工具类的细节完善起来,形成jar包供自己或者供开源。

相关连接

这只是后台对结果的统一处理,我另外写了一篇文章:JSON工具类的构建(前端版本),配合使用效果更佳哦~

本文首发于cartoon的博客
转载请注明出处:https://cartoonyu.github.io/cartoon-blog/post/json/json工具类的构建后端版本/

JSON工具类的构建(后端版本)的更多相关文章

  1. JSON工具类的构建(前端版本)

    前言 在前后端交互的选择上,之前一直采用的是模板引擎(因为我只负责后端). 而这次的一个算是作业吧,前后端都是我,所以就研究了一下JSON交互在java web的应用(主要是前端). 优缺点 因为我是 ...

  2. Spring统一返回Json工具类,带分页信息

    前言: 项目做前后端分离时,我们会经常提供Json数据给前端,如果有一个统一的Json格式返回工具类,那么将大大提高开发效率和减低沟通成本. 此Json响应工具类,支持带分页信息,支持泛型,支持Htt ...

  3. 用jackson封装的JSON工具类

    package hjp.smart4j.framework.util; import com.fasterxml.jackson.databind.ObjectMapper; import org.s ...

  4. Code片段 : .properties属性文件操作工具类 & JSON工具类

    摘要: 原创出处:www.bysocket.com 泥瓦匠BYSocket 希望转载,保留摘要,谢谢! “贵专” — 泥瓦匠 一.java.util.Properties API & 案例 j ...

  5. Json工具类,实现了反射将整个Object转换为Json对象的功能,支持Hibernate的延迟加

    package com.aherp.framework.util; import java.lang.reflect.Array;import java.lang.reflect.Method;imp ...

  6. Json工具类 - JsonUtils.java

    Json工具类,提供Json与对象之间的转换. 源码如下:(点击下载 - JsonUtils.java . gson-2.2.4.jar ) import java.lang.reflect.Type ...

  7. Java json工具类,jackson工具类,ObjectMapper工具类

    Java json工具类,jackson工具类,ObjectMapper工具类 >>>>>>>>>>>>>>> ...

  8. 强大的Java Json工具类

    转自: https://blog.csdn.net/u014676619/article/details/49624165 import java.io.BufferedReader; import ...

  9. HttpClientUntils工具类的使用测试及注意事项(包括我改进的工具类和Controller端的注意事项【附 Json 工具类】)

    HttpClient工具类(我改过): package com.taotao.httpclient; import java.io.IOException; import java.net.URI; ...

随机推荐

  1. Cocos2d 之FlyBird开发---GamePlay类

    |   版权声明:本文为博主原创文章,未经博主允许不得转载. 这个是游戏的核心部分:(FlyBird游戏重中之重) 创建一个物理世界(世界设置重力加速度) 在物理世界中添加一个动态的刚体(小鸟) 在物 ...

  2. if语句基本练习需求

    1.需求:键盘录入一个成绩,判断并输出成绩的等级. 90-100 优 80-89 良好 70-79 中等 60-69 及格 0-59 不及格 import java.util.Scanner; cla ...

  3. LeetCode6 dp

    120. Triangle 我的解法用了一个二维数组,这样比较浪费空间.O(n*n) 但是标准答案自底向上,一是不需要进行特别判断,二是可以覆盖数组,则只需要O(n)的空间大小. class Solu ...

  4. 想要成长,总得立个flag

    国庆假期结束了,自己的成长却丝毫没有,所以啊,立个flag试试看,每天能发表一个小知识点,希望自己能够做到,加油

  5. js五种基本数据类型:string, number, boolean, null, undefined

    /** * 五种基本数据类型:string, number, boolean, null, undefined */ // undefined // 声明变量foo,未声明变量bar var foo; ...

  6. win10下安装Ubuntu后,启动时没有win10选项解决方法

    通过在ubuntu里修改启动引导,解决. 1.进入Ubuntu系统,Ctrl+alt+t进入终端,输入以下命令即可 sudo gedit /etc/default/grub 2.在打开的gedit编辑 ...

  7. 分支结构if 语句举例

  8. dotNET面试(一)

    1.列举ASP.NET 页面之间传递值的几种方式. 1).使用QueryString, 如....?id=1; response. Redirect().... 2).使用Session变量 3).使 ...

  9. 解决mac pro 软件损坏

    1,打开终端 2,输入 sudo spctl --master-disable 3,打开系统偏好设置——>安全与隐私——>勾选任何来源

  10. Bootstrap 网页1

    <!DOCTYPE html> <html> <head> <meta charset="utf-8" /> <title&g ...