本文介绍json的规范及javascript和java对数据的交换读取
- 1. json介绍
- 1.1 json简介
- 1.2为什么使用 JSON?
 
- 2. json规范
- 2.1基础规范
- 2.2 key值为-字符串、数字、布尔值
- 2.3 key值为对象Object
- 2.4 key值为数组
- 2.5 json本身就是一个数组
 
- 3.javascript操作json
- 3.1 javascript操作json对象
- 3.2 js操作json字符串,把字符串转化为json对象
- 3.3 键值为null或者键不存在判断
- 3.4json对象转字符串
 
- 4.java操作json
- 4.1准备两个比较全面的VO类
- 4.1 FastJson介绍
- 4.1.1引入依赖
- 4.1.2 vo基础解析
- 4.1.3 把null的字段全部传递过来,值显示为null
- 4.1.4 字符串类型的null处理成""
- 4.1.5 数字、布尔、数组/list、日期null处理
- 4.1.5 把null全部转化成""
- 4.1.6日期转换
- 4.1.7 map处理
- 4.1.8 list处理
- 4.1.9 总结
- 4.1.10 json转java对象
 
- 4.2 Gson介绍
- 4.2.1引入依赖
- 4.2.2 vo基础解析
- 4.2.3 把值为null的传递出去
- 4.2.4 toJson支持对象所有属性转化
- 4.2.5处理null-空值
- 4.2.6 日期处理
- 4.2.7 list和 map处理
- 4.2.8 JSON转java对象
 
- 4.3 Jackson介绍
- 4.3.1引入依赖
- 4.3.2 vo基础解析
- 4.3.3全局配置
- 4.3.4 支持vo里所有复杂属性的转化
- 4.4.5 把null全部转化成""
- 4.4.6日期处理
- 4.4.7 list处理
- 4.4.8 map处理
- 4.3.9 json转vo
 
- 4.4解析总结
- 4.5调调用第三方接口json解析
 
1. json介绍
1.1 json简介
JSON(JavaScript Object Notation, JS对象简谱)是一种轻量级的数据交换格式。
 它基于 ECMAScript(European Computer Manufacturers Association, 欧洲计算机协会制定的js规范)的一个子集,采用完全独立于编程语言的文本格式来存储和表示数据。
 简洁和清晰的层次结构使得 JSON 成为理想的数据交换语言。
 易于人阅读和编写,同时也易于机器解析和生成,并有效地提升网络传输效率。
1.2为什么使用 JSON?
对于 AJAX 应用程序来说,JSON 比 XML 更快更易使用:
 使用 XML
- 读取 XML 文档
- 使用 XML DOM 来循环遍历文档
- 读取值并存储在变量中
使用 JSON
- 读取 JSON 字符串
- 用 eval() 处理 JSON 字符串
2. json规范
2.1基础规范
eg:
{
"name":"蒋增奎","address":"天府大道","age":30, "isMoney":true
}
规范说明:
- json内容需要用花括号{}包裹起来
- 键值对用:间隔,多个键,中间用逗号隔离,最后一个键不需要逗号
- 键key如果没有特殊符号,可以不用双引号包围。
- 值的类型,包括字符串、数字、布尔值、对象、数组五种类型,前三种最常见
key可以不需要双引号如:
2.2 key值为-字符串、数字、布尔值
{
name:"蒋增奎",address:"天府大道",age:30, isMoney:true
}
2.3 key值为对象Object
嵌套就是json对象里面又包含对象,key对应的值,除了括字符串、数字、布尔值三种类型外,还可以是一个对象,其规范是
key:{对象属性}
  var obj4 = {
          name: "雪邦科技",
          boss: {
            name: "蒋增奎--",
            age: 40,
          },
        };
2.4 key值为数组
数组用方括号包围一起来[x1,x2,x3],分组之间用逗号隔离
数组值有两种类型:
1.基本类型:如[1,2,3]
2.json对象,json对象要用{}包裹
//---------------------------------------------
 var obj2={
 		    wifes: ["冰冰", "雅致"],  //基础类型,对应后端是一个基础类型的数组或者list
      
            "user":[ //是一个对象,对应后都是一个对象的数组或者List
                    {"name":"jzk","sex":20},
                    {"name":"jj","sex":30},
                    ],
            "address":[
                       {"shen":"云南","city":"昆明"},
                       {"shen":"四川","city":"雅安"},
                       
                       ]
    };
//---------------------------------------------
 var obj3={
 "name":"雪邦科技",
 "address":"天府大道",
"employees": [
	{ "firstName":"Bill" , "lastName":"Gates" },
	{ "firstName":"George" , "lastName":"Bush" },
	{ "firstName":"Thomas" , "lastName":"Carter" }
]
}
2.5 json本身就是一个数组
有两种类型:
1.基本类型数组:[3,2,1]
2.对象的数组:[{x1},{x2},{x3},] 对象要用{}包括
var obj1=[
              {"name":"蒋增奎","address":"天府大道","sex":30},
              {"name":"奎奎","address":"大道","sex":38},
          ];
var obj2=[1,2,3];
3.javascript操作json
语法:
- 把json赋值一个给一本变量,通过变量直接用对象名.key就可以读取
- 如果是数组,则可以循环,obj[i]读取
- 如果key值为对象,直接用:对象名.key.属性读取
- 如果要修改值,直接用对象名.key=xxx则可以修改
3.1 javascript操作json对象
eg:
 <script>
      function dd() {
        //1.基本属性读取=========================
        var obj = {
          name: "蒋增奎1",
          address: "天府大道",
          sex: 30,
          exist: true,
        };
        alert(obj.name);
        obj.name = "jzk"; //改变属性值
        //2.key值为对象==========================
        var obj4 = {
          name: "雪邦科技",
          boss: {
            name: "蒋增奎--",
            age: 40,
          },
        };
        alert("obj4.boss.name=" + obj4.boss.name);
        //3.json本身是一个数组====================
        var obj1 = [
          { name: "蒋增奎", address: "天府大道", sex: 30 },
          { name: "奎奎", address: "大道", sex: 38 },
        ];
        //循环读取
        for (index in obj1) {
          alert(
            "数组[" +
              index +
              "]=" +
              obj1[index].name +
              "-" +
              obj1[index].address +
              "-" +
              obj1[index].sex
          );
        }
        //4.key值是一个数组
        var company = {
          name: "雪邦",
          address: "天府大道1号",
          boss: "蒋增奎",
          exist: true,
          tax: 5000,
          users: [
            { name: "蒋增奎", sex: "男", mobile: "13688006645" },
            { name: "张三", sex: "女", mobile: "13688006640" },
          ],
        };
        var users = company.users;
        for (index in users) {
          alert(
            "company.users数组[" +
              index +
              "]=" +
              users[index].name +
              "-" +
              users[index].mobile +
              "-" +
              users[index].sex
          );
        }
      
      }
    </script>
3.2 js操作json字符串,把字符串转化为json对象
有两种写法:
(1)JSON.parse(字符串)
(2) eval (“(” + txt + “)”);
eg:
   <script>
      function dd() {
        //1.eval((json字符串))语法转换成json对象
        var s = '{"name":"蒋增奎","address":"天府大道","sex":30}';
        s = "(" + s + ")";
        var obj5 = eval(s);
         alert(obj5.sex);
        // 2.JSON.parse(json字符串)语法转换成json对象
        var txt =
          '{ "sites" : [' +
          '{ "name":"菜鸟教程" , "url":"www.runoob.com" },' +
          '{ "name":"google" , "url":"www.google.com" },' +
          '{ "name":"微博" , "url":"www.weibo.com" } ]}';
        var obj6 = JSON.parse(txt);
        alert(obj6.sites[0].name);
      }
    </script>
3.3 键值为null或者键不存在判断
如果键值为null.可以直接用:obj.sex == null判断
如果键不存在,用:typeof obj.deptId == "undefined"判断
function dd2() {
        var obj = {
          name: "蒋增奎1",
          address: "天府大道",
          sex: null,
          exist: true,
          deptId: 3,
        };
        //如果获取得一个键key本身不存在
        if (!obj.deptId) {
          alert("不存在obj.deptId");
        }
        if (obj.deptId == null) {
          alert("不存在obj.deptId");
        }
        alert(obj.deptId);
        //存在key,但值为null
        if (!obj.sex) {
          alert("不存在obj.sex");
        }
        //存在key,但值为null
        if (obj.sex == null) {
          alert("obj.sex==null");
        }
        alert(obj.sex);
        //通过typeof可以判断出是否是null还是key不存在
        if (typeof obj.deptId == "undefined") {
          alert("key==deptId不存在");
        }
        //排除掉无key和key值为null
        if (typeof obj.deptId != "undefined" && obj.deptId) {
          alert("key==deptId存在并且值不为null");
        }
      }
3.4json对象转字符串
var str=JSON.stringify(json对象)
  function dd1() {
        var obj = { name: "runoob", alexa: 10000, site: "www.runoob.com" };
        var JSONStr = JSON.stringify(obj);
        alert(JSONStr);
      }
4.java操作json
Java中并没有内置JSON的解析,因此使用JSON需要借助第三方类库。
 下面是几个常用的 JSON 解析类库:
- Gson: 谷歌开发的 JSON 库,功能十分全面。
- FastJson: 阿里巴巴开发的 JSON 库,性能十分优秀。
- Jackson: 社区十分活跃且更新速度很快。,springMVC默认
4.1准备两个比较全面的VO类
@Data
@NoArgsConstructor
@AllArgsConstructor
public class JsonVO {
    private Long id;
    private String name;
    private BigDecimal weight;//年纪
    private Date birth;
    private Date creatTime;
    private Boolean money;
    private Address address;
    private List<Address> addresses;
    private Map<String,Object> son;
    private String[] wifes;
}
@Data
@NoArgsConstructor
@AllArgsConstructor
public class Address {
    private String shen;
    private String  city;
}
4.1 FastJson介绍
4.1.1引入依赖
 <dependency>
            <groupId>com.alibaba</groupId>
            <artifactId>fastjson</artifactId>
            <version>1.2.47</version>
 </dependency>
4.1.2 vo基础解析
最基础,直接调用JSON.toJSONString(vo);
/**
* 问题:
* 1.vo对象属性为null不会传递
* 2.日期转化成时间戳数字
*
*/
    @Test
    public void test2(){
        //--------------vo转json-------------
        JsonVO vo=new JsonVO();
        vo.setId(1l);
        vo.setBirth(new Date());
        String jsonString = JSON.toJSONString(vo);
        System.out.println("只传vo对象:"+jsonString);
        
    }
打印效果
只传vo对象:{"birth":1701706009427,"id":1}
4.1.3 把null的字段全部传递过来,值显示为null
传入参数:SerializerFeature.WriteMapNullValue
问题:前端需要去处理null
 @Test
    public void test3(){
        JsonVO vo=new JsonVO();
        vo.setId(1l);
        vo.setBirth(new Date());
        String jsonString = "";
        jsonString = JSON.toJSONString(vo,
                SerializerFeature.WriteMapNullValue
        );
        System.out.println(jsonString);
    }
效果
{"addresses":null,"birth":1701707310898,"creatTime":null,"id":1,
"money":null,"name":null,"weight":null}
4.1.4 字符串类型的null处理成""
null的字符串类型能传递,并转化成""
问题:
1.日期和自定义对象依然显示为nul
2.数组、list、数字、boolean无法传递
@Test
    public void test4(){
        JsonVO vo=new JsonVO();
        vo.setId(1l);
        vo.setBirth(new Date());
        String jsonString = "";
        jsonString = JSON.toJSONString(vo,
                SerializerFeature.WriteNullStringAsEmpty    //字符串为null的转化为""
        );
        System.out.println(jsonString);
    }
效果
{"address":null,"birth":1701708039585,"creatTime":null,"id":1,"name":""}
4.1.5 数字、布尔、数组/list、日期null处理
SerializerFeature.WriteNullStringAsEmpty, //字符串为null的转化为""
SerializerFeature.WriteNullNumberAsZero, //数字为null转化为0
SerializerFeature.WriteNullListAsEmpty, //数组、list为null的转化成[]
SerializerFeature.WriteNullBooleanAsFalse,//布尔值为null转化成false
SerializerFeature.WriteDateUseDateFormat //日期把时间戳格式化
问题
1.日期、自定义的对象依然为null
2.如果数字0,和布尔值false有意义,前端区分不出来
3.日期区分不出来是时间还是日期
 @Test
    public void test5(){
        JsonVO vo=new JsonVO();
        vo.setId(1l);
        vo.setBirth(new Date());
        String jsonString = "";
        jsonString = JSON.toJSONString(vo,
                SerializerFeature.WriteNullStringAsEmpty,    //字符串为null的转化为""
                SerializerFeature.WriteNullNumberAsZero,   //数字为null转化为0
                SerializerFeature.WriteNullListAsEmpty,   //数组、list为null的转化成[]
                SerializerFeature.WriteNullBooleanAsFalse,//布尔值为null转化成false
                SerializerFeature.WriteDateUseDateFormat   //日期把时间戳格式化
        );
        System.out.println(jsonString);
    }
效果
{"address":null,"addresses":[],"birth":"2023-12-05 00:51:55",
"creatTime":null,"id":1,"money":false,"name":"","weight":0}
4.1.5 把null全部转化成""
重写过滤器,拦截null的,全部置换成""
问题:
1.前端如果有数字计算,要去转化
2.如果自定义对象,数组等,需要先判断""再做相关操作
过滤器
public class JsonUtil {
    /**
     * 过滤器,把null全部转化成空字符串
     * @return
     */
    public static ValueFilter  getFilter(){
        ValueFilter filter = new ValueFilter() {
            @Override
            public Object process(Object obj, String s, Object v) {
               // System.out.println(s);
                if(v==null)
                    return "";
                return v;
            }
        };
        return filter;
    }
}
解析
{
"address":"","addresses":"","birth":1701709403555,"creatTime":"","id":1,
"money":"","name":"","weight":""
}
4.1.6日期转换
采用注解的方式,可以解析出日期和时间两种,此方法最完美
在日期变量属性上加:
@JSONField(format=“yyyy-MM-dd”)或者
@JSONField(format=“yyyy-MM-dd HH:mm:ss”)
vo
@Data
@NoArgsConstructor
@AllArgsConstructor
public class JsonVO {
    private Long id;
    private String name;
    private BigDecimal weight;//年纪
    @JSONField(format="yyyy-MM-dd")
    private Date birth;
    @JSONField(format="yyyy-MM-dd HH:mm:ss")
    private Date creatTime;
    private Boolean money;
    private Address address;
    private List<Address> addresses;
}
测试
  @Test
    public void test7(){
        JsonVO vo=new JsonVO();
        vo.setId(1l);
        vo.setBirth(new Date());
        vo.setCreatTime(new Date());
        String jsonString = "";
        jsonString = JSON.toJSONString(vo);
        System.out.println(jsonString);
    }
效果
{"birth":"2023-12-05","creatTime":"2023-12-05 01:07:51","id":1}
4.1.7 map处理
@Test
  @Test
    public void test8(){
        Map<String,Object> map=new HashMap<String,Object>();
        map.put("name","jzk");
        map.put("id",1);
        map.put("sex",null);
        String jsonString = "";
        jsonString = JSON.toJSONString(map,JsonUtil.getFilter());
        System.out.println(jsonString);
    }
效果:
{"sex":"","name":"jzk","id":1}
4.1.8 list处理
不需要特殊处理
 @Test
    public void test9(){
        List<Address> list=new ArrayList<>();
        list.add(new Address("四川","成都"));
        list.add(new Address("四川","简阳"));
        list.add(new Address());
        String jsonString = "";
        jsonString = JSON.toJSONString(list,JsonUtil.getFilter());
        System.out.println(jsonString);
    }
效果:
[{"city":"成都","shen":"四川"},{"city":"简阳","shen":"四川"},{"city":"","shen":""}]
4.1.9 总结
用过滤器还是最简单
4.1.10 json转java对象
采用语法VO vo=JSON.parseObject(jsonStr,VO.class);
如果key是"",其他对象可以转化成null
这里有两个特殊的:
- 数组转化成list
要用JSON.parseArray来转化,JSONArray 本身也是一个数组
2.数组转化成map
| 方法 | 说明 | 
|---|---|
| SON.parseObject(jsonStgr,映射类.class); | json和一个类对应,不管这个vo类里属性有多复杂,都能映射上 | 
| Map<String,Object> map=JSON.parseObject(jsonStgr,Map.class); | 和map对象的映射 | 
| JSONArray jsonArray = JSONArray.parseArray(jsonStgr); | 返回一个数组,json本身也是一个数组 | 
| List< VO > list=JSON.parseArray(jsonStgr,VO.class); | 映射到一个List< VO > 列表里 | 
  @Test
    public void test33(){
       String txt="{'address':'','addresses':'','birth':'2023-12-05','creatTime':'','id':1,'money':'','name':'','weight':''}";
       JsonVO vo= JSON.parseObject(txt,JsonVO.class);
        System.out.println(vo);
        //json有,但vo没有的字段,不报错,
        //waihao这个属性,vo对象里没有
        txt="{'name':'jzk','waihao':'李富贵'}";
        vo= JSON.parseObject(txt,JsonVO.class);
        System.out.println(vo);
    
        //自定义对象
        txt="{'address':{'city':'cd','shen':'sc'},'addresses':'','birth':'2023-12-05','creatTime':'','id':1,'money':'','name':'','weight':''}";
        vo= JSON.parseObject(txt,JsonVO.class);
        //数组
        txt="{'address':'','addresses':[{'city':'cd','shen':'sc'},{'city':'my','shen':'sc'}],'birth':'2023-12-05','creatTime':'','id':1,'money':'','name':'','weight':''}";
        vo= JSON.parseObject(txt,JsonVO.class);
        System.out.println(vo);
 //数组
        txt="{'address':'','addresses':[{'city':'cd','shen':'sc'},{'city':'my','shen':'sc'}],'birth':'2023-12-05','creatTime':'','id':1,'money':'','name':'','weight':''}";
        vo= JSON.parseObject(txt,JsonVO.class);
        System.out.println(vo);
        
        //-------------------特殊数据类型---------------------
        //纯数组转list
        txt="[{'city':'成都','shen':'四川'},{'city':'简阳','shen':'四川'},{'city':'','shen':''}]";
        List<Address> list=JSON.parseArray(txt,Address.class);
        for (Address a:list){
            System.out.println(a);
        }
        //数组转数组,数组转数组没有直接方法,需要用JSONArray来过渡
        System.out.println("数组转数组");
        txt="[{'city':'成都','shen':'四川'},{'city':'简阳','shen':'四川'},{'city':'','shen':''}]";
        JSONArray jsonArray = JSONArray.parseArray(txt);
        Address[] ss=new Address[jsonArray.size()];
        for(int i=0;i<jsonArray.size();i++){
            Address address=jsonArray.getObject(i,Address.class);
            ss[i]=address;
        }
        for (Address address:ss){
            System.out.println(address);
        }
        //str转map
        txt="{'city':'成都','shen':'四川'}";
        Map<String,Object> map=JSON.parseObject(txt,Map.class);
        for(String key:map.keySet()){
            String value = map.get(key).toString();
            System.out.println("key="+key+" vlaue="+value);
        }
    }
4.2 Gson介绍
4.2.1引入依赖
  <dependency>
            <groupId>com.google.code.gson</groupId>
            <artifactId>gson</artifactId>
            <version>2.8.5</version>
        </dependency>
    </dependencies>
4.2.2 vo基础解析
最基础,直接调
Gson gson = new Gson();
String jsonStr=gson.toJson(vo,JsonVO.class);
/**
* 问题:
* 1.vo对象属性为null不会传递
* 2.日期转化成没有格式化
*
*/
@Test
    public void t1(){
        Gson gson = new Gson();
        JsonVO vo=new JsonVO();
        vo.setName("jzk");
        vo.setBirth(new Date());
        String jsonStr=gson.toJson(vo,JsonVO.class);
        System.out.println(jsonStr);
    }
效果
{"name":"jzk","birth":"Dec 6, 2023 9:51:56 PM"}
4.2.3 把值为null的传递出去
这里是GsonBuilder来创建Gson对象
Gson gson = new GsonBuilder().serializeNulls().create();
这样获得全部null
@Test
    public void t2(){
        Gson gson = new GsonBuilder().serializeNulls().create();
        JsonVO vo=new JsonVO();
       // vo.setName("jzk");
        vo.setId(2l);
        vo.setBirth(new Date());
        String  jsonStr=gson.toJson(vo,JsonVO.class);
        System.out.println(jsonStr);
    }
效果
{
"id":2,"name":null,"weight":null,"birth":"Dec 6, 2023 9:54:19 PM",
"creatTime":null,"money":null,"address":null,
"addresses":null,"son":null,"wifes":null
}
4.2.4 toJson支持对象所有属性转化
public JsonVO getAllVo(){
        JsonVO vo=new JsonVO();
        vo.setName("jzk");
        vo.setBirth(new Date());
        vo.setId(1l);
        vo.setCreatTime(new Date());
        vo.setMoney(true);
        vo.setWeight(new BigDecimal("20.56"));
        vo.setAddress(new Address("四川","成都"));
        List<Address> list=new ArrayList<Address>();
        list.add(new Address("四川","绵阳"));
        list.add(new Address("四川","简阳"));
        vo.setAddresses(list);
        Map<String,Object> son=new HashMap<>();
        son.put("name","将填好");
        son.put("sex","1");
        vo.setSon(son);
        vo.setWifes(new String[]{"冰冰","雅致"});
        return vo;
    }
    @Test
    public void t3(){
        Gson gson = new Gson();
        JsonVO vo=getAllVo();
        String  jsonStr=gson.toJson(vo,JsonVO.class);
        System.out.println(jsonStr);
    }
效果
{
"id":1,"name":"jzk","weight":20.56,"birth":"Dec 6, 2023 9:57:35 PM",
"creatTime":"Dec 6, 2023 9:57:35 PM","money":true,
"address":{"shen":"四川","city":"成都"},
"addresses":
[
{"shen":"四川","city":"绵阳"},{"shen":"四川","city":"简阳"}
],
"son":{"sex":"1","name":"将填好"},
"wifes":["冰冰","雅致"]
}
4.2.5处理null-空值
GsonBuilder.registerTypeAdapter注册,进行重写
帮助类
public class GsonUtil {
    public static Gson getGson(){
        Gson gson =new  GsonBuilder().serializeNulls()   //null全部显示
                .registerTypeAdapter(String.class,new StringAdapter())  //空字符串处理""
                .registerTypeAdapter(Long.class,new LongAdapter())  //Long字段处理成""
                .registerTypeAdapter(Integer.class,new IntegerAdapter()) //Integer
                .registerTypeAdapter(BigDecimal.class,new BigDecimalAdapter()) //BigDecimal
                .registerTypeAdapter(Date.class,new DateAdapter()) //Date
                .registerTypeAdapter(Boolean.class,new BooleanAdapter())//Boolean
                .create();
        return gson;
    }
    /**
     * 字符串为""
     */
    public static class StringAdapter extends TypeAdapter<String> {
        @Override
        public void write(JsonWriter jsonWriter, String s) throws IOException {
            if (s == null) {//序列化使用的是adapter的write方法
                //jsonWriter.nullValue();//这个方法是错的,而是应该将null转成""
                jsonWriter.value("");
                return;
            }
            jsonWriter.value(s);
        }
        @Override
        public String read(JsonReader jsonReader) throws IOException {
            if (jsonReader.peek() == JsonToken.NULL) {//反序列化使用的是read方法
                jsonReader.nextNull();
                return null;
            }
            return jsonReader.nextString();
        }
    }
    /**
     * Long null为""
     */
    public static class LongAdapter extends TypeAdapter<Long> {
        @Override
        public void write(JsonWriter jsonWriter, Long s) throws IOException {
            if (s == null) {//序列化使用的是adapter的write方法
                //jsonWriter.nullValue();//这个方法是错的,而是应该将null转成""
                jsonWriter.value("");
                return;
            }
            jsonWriter.value(s);
        }
        @Override
        public Long read(JsonReader jsonReader) throws IOException {
            if (jsonReader.peek() == JsonToken.NULL) {//反序列化使用的是read方法
                jsonReader.nextNull();
                return null;
            }
            return jsonReader.nextLong();
        }
    }
    /**
     * Double
     */
    public static class DoubleAdapter extends TypeAdapter<Double> {
        @Override
        public void write(JsonWriter jsonWriter, Double s) throws IOException {
            if (s == null) {//序列化使用的是adapter的write方法
                //jsonWriter.nullValue();//这个方法是错的,而是应该将null转成""
                jsonWriter.value("");
                return;
            }
            jsonWriter.value(s);
        }
        @Override
        public Double read(JsonReader jsonReader) throws IOException {
            if (jsonReader.peek() == JsonToken.NULL) {//反序列化使用的是read方法
                jsonReader.nextNull();
                return null;
            }
            return jsonReader.nextDouble();
        }
    }
    /**
     * Boolean
     */
    public static class BooleanAdapter extends TypeAdapter<Boolean> {
        @Override
        public void write(JsonWriter jsonWriter, Boolean s) throws IOException {
            if (s == null) {//序列化使用的是adapter的write方法
                //jsonWriter.nullValue();//这个方法是错的,而是应该将null转成""
                jsonWriter.value("");
                return;
            }
            jsonWriter.value(s);
        }
        @Override
        public Boolean read(JsonReader jsonReader) throws IOException {
            if (jsonReader.peek() == JsonToken.NULL) {//反序列化使用的是read方法
                jsonReader.nextNull();
                return null;
            }
            return jsonReader.nextBoolean();
        }
    }
    /**
     * BigDecimal
     */
    public static class BigDecimalAdapter extends TypeAdapter<BigDecimal> {
        @Override
        public void write(JsonWriter jsonWriter, BigDecimal s) throws IOException {
            if (s == null) {//序列化使用的是adapter的write方法
                //jsonWriter.nullValue();//这个方法是错的,而是应该将null转成""
                jsonWriter.value("");
                return;
            }
            jsonWriter.value(s);
        }
        @Override
        public BigDecimal read(JsonReader jsonReader) throws IOException {
            if (jsonReader.peek() == JsonToken.NULL) {//反序列化使用的是read方法
                jsonReader.nextNull();
                return null;
            }
            return new BigDecimal(jsonReader.nextString());
        }
    }
    /**
     * Integer
     */
    public static class IntegerAdapter extends TypeAdapter<Integer> {
        @Override
        public void write(JsonWriter jsonWriter, Integer s) throws IOException {
            if (s == null) {//序列化使用的是adapter的write方法
                //jsonWriter.nullValue();//这个方法是错的,而是应该将null转成""
                jsonWriter.value("");
                return;
            }
            jsonWriter.value(s);
        }
        @Override
        public Integer read(JsonReader jsonReader) throws IOException {
            if (jsonReader.peek() == JsonToken.NULL) {//反序列化使用的是read方法
                jsonReader.nextNull();
                return  null;
            }
            return jsonReader.nextInt();
        }
    }
    /**
     * Integer
     */
    public static class DateAdapter extends TypeAdapter<Date> {
        @Override
        public void write(JsonWriter jsonWriter, Date s) throws IOException {
            if (s == null) {//序列化使用的是adapter的write方法
                //jsonWriter.nullValue();//这个方法是错的,而是应该将null转成""
                jsonWriter.value("");
                return;
            }
            String temp=getStrByDate(s,"yyy-MM-dd HH:mm:ss");
            if(temp.indexOf("00:00:00")!=-1){
                temp=temp.substring(0,temp.indexOf("00:00:00"));
                temp= temp.trim();
            }
            jsonWriter.value(temp);
        }
        @Override
        public Date read(JsonReader jsonReader) throws IOException {
            if (jsonReader.peek() == JsonToken.NULL) {//反序列化使用的是read方法
                jsonReader.nextNull();
                return null;
            }
          //  return jsonReader.nextString();
            System.out.println("jsonReader.nextString()="+jsonReader.nextString());
            return getDateByStr(jsonReader.nextString(),"yyy-MM-dd HH:mm:ss");
        }
    }
    private static String getStrByDate(Date date,String gs){
        if(date==null)
            return "";
        DateFormat df = new SimpleDateFormat(gs);
        return  df.format(date);
    }
    private static Date getDateByStr(String dateStr,String gs){
        if(dateStr==null || dateStr.trim().length()==0)
            return null;
        DateFormat df = new SimpleDateFormat(gs);
        try{
            return df.parse(dateStr);
        }
        catch(ParseException e){
            throw new RuntimeException("日期转换出错:"+e);
        }
    }
}
代码
@Test
    public void t4(){
        Gson gson= GsonUtil.getGson();
        //然后用上面一行写的gson来序列化和反序列化实体类type
        JsonVO vo=new JsonVO();
        // vo.setName("jzk");
       // vo.setId(2l);
       //设置一个没有时间的日期
        vo.setBirth(DateUtil.getDateByStr("2022-10-01",DateUtil.date_gs));
        String  jsonStr=gson.toJson(vo,JsonVO.class);
        System.out.println(jsonStr);
    }
效果
{"id":"","name":"","weight":"","birth":"2022-10-01",
"creatTime":"","money":"","address":null,"addresses":null,
"son":null,"wifes":null,"birth2":null}
4.2.6 日期处理
有三种方式:
1.GsonBuilder…setDateFormat(“yyyy-MM-dd HH:mm:ss”) //设置日期转换
问题:对象里的所有日期只能用一种格式
2.自定义一个类日期属性的注解【如FastJson,Gson没有自带日期注解】
3.做一个TypeAdapter的实现类,自己写(看上文GsonUtil.java里面的代码)
@Test
    public void t6(){
        //通过setDateFormat来设置日期格式,问题:所有数据都设置成了一种格式
        //gson没有提供注解,可以自定义一个注解,处理VO对象里有日期和时间两个字段
        Gson gson  = new GsonBuilder()
                .setDateFormat("yyyy-MM-dd HH:mm:ss") //设置日期转换
                .create();
        JsonVO vo=new JsonVO();
        // vo.setName("jzk");
        // vo.setId(2l);
        vo.setBirth(new Date());
        vo.setCreatTime(new Date());
        String  jsonStr=gson.toJson(vo,JsonVO.class);
        System.out.println(jsonStr);
    }
效果:
{"birth":"2023-12-06 23:57:12","creatTime":"2023-12-06 23:57:12"}
4.2.7 list和 map处理
gson.toJson(list,ArrayList.class);
gson.toJson(map,HashMap.class);
 @Test
    public void t33(){
        Gson gson = new Gson();
        List<Address> list=new ArrayList<Address>();
        list.add(new Address("四川","绵阳"));
        list.add(new Address("四川","简阳"));
       String  jsonStr=gson.toJson(list,ArrayList.class);
        System.out.println(jsonStr);
        Map<String,Object> map=new HashMap();
        map.put("name","jzk");
        map.put("address","天赋达到");
        jsonStr=gson.toJson(map,HashMap.class);
        System.out.println(jsonStr);
    }
效果
[{"shen":"四川","city":"绵阳"},{"shen":"四川","city":"简阳"}]
{"address":"天赋达到","name":"jzk"}
4.2.8 JSON转java对象
1.json转vo对象
JsonVO vo=new Gson().fromJson(jsonStr,JsonVO.class);
2.json转list< vo >
Type type = new TypeToken<List< Address>>(){}.getType();//获取转化类型
List< Address> list=new Gson().fromJson(json,type)
3.json转map
Map<String,String> map = new Gson().fromJson(json, Map.class);
@Test
    public void toVo(){
        //有null的数据
        String json="{'id':2,'name':null,'weight':null,'birth':'2023-10-23','creatTime':null,'money':null,'address':null,'addresses':null,'son':null,'wifes':null}";
        JsonVO vo=new Gson().fromJson(json,JsonVO.class);
      System.out.println(vo);
      //所有数据都有值
json="{'id':1,'name':'jzk','weight':20.56,'birth':'2021-10-11','creatTime':'2023-10-01 12:23:23','money':true,'address':{'shen':'四川','city':'成都'},'addresses':[{'shen':'四川','city':'绵阳'},{'shen':'四川','city':'简阳'}],'son':{'sex':'1','name':'将填好'},'wifes':['冰冰','雅致']}";
      vo =new Gson().fromJson(json,JsonVO.class);
      System.out.println(vo);
      //json有,vo对象没有字段,不报错
      json="{'id':2,'waihao':'黑旋风'}";  //waihao这个属性,vo没有
       vo=new Gson().fromJson(json,JsonVO.class);
      System.out.println(vo);
      //纯List转换
        json="[{'shen':'四川','city':'绵阳'},{'shen':'四川','city':'简阳'}]";
        Type type = new TypeToken<List<Address>>(){}.getType();//获取转化类型
        List<Address> list=new Gson().fromJson(json,type);
        for(Address a:list){
             System.out.println(a);
        }
      //map转换
      json="{'shen':'四川','city':'绵阳'}";
      Map<String,String> map = new Gson().fromJson(json, Map.class);
      for(String key:map.keySet()){
          String value = map.get(key).toString();
          System.out.println("key="+key+" vlaue="+value);
      }
  }
4.3 Jackson介绍
4.3.1引入依赖
  <dependency>
            <groupId>com.fasterxml.jackson.core</groupId>
            <artifactId>jackson-databind</artifactId>
            <version>2.9.1</version>
 </dependency>
4.3.2 vo基础解析
String str=new ObjectMapper()…writeValueAsString(解析对象);
问题:
1.全部null都会传递过来
2.时间展示是时间戳格式
 @Test
    public void t1() throws JsonProcessingException {
        JsonVO vo=new JsonVO();
        vo.setId(2l);
        vo.setBirth(new Date());
        ObjectMapper mapper=new ObjectMapper();
        String jsonStr="";
        jsonStr=mapper.writeValueAsString(vo);
        System.out.println(jsonStr);
    }
效果
{"id":2,"name":null,"weight":null,"birth":1701912127118,"creatTime":null,
"money":null,"address":null,"addresses":null,"son":null,
"wifes":null,"birth2":null}
4.3.3全局配置
//在反序列化时忽略在 json 中存在但 Java 对象不存在的属性
mapper.configure(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES, false);
//在序列化时日期格式默认为 yyyy-MM-dd’T’HH:mm:ss.SSSZ
mapper.configure(SerializationFeature.WRITE_DATES_AS_TIMESTAMPS, false);
//在序列化时自定义时间日期格式
mapper.setDateFormat(new SimpleDateFormat(“yyyy-MM-dd HH:mm:ss”));
//在序列化时忽略值为 null 的属性
mapper.setSerializationInclusion(JsonInclude.Include.NON_NULL);
//在序列化时忽略值为默认值的属性
mapper.setDefaultPropertyInclusion(JsonInclude.Include.NON_DEFAULT);
//在反序列时,配置允许单引号,否则json里面的单引号如: ‘name’:'jzk’会报错
mapper.configure(JsonParser.Feature.ALLOW_SINGLE_QUOTES, true);
//输出json格式化,调试的时候可使用,增加易读性
mapper.writerWithDefaultPrettyPrinter()
4.3.4 支持vo里所有复杂属性的转化
@Test
    public void t2()throws JsonProcessingException {
        ObjectMapper mapper=new ObjectMapper();
        JsonVO vo=getAllVo();
        String jsonStr="";
        jsonStr=mapper.writerWithDefaultPrettyPrinter().writeValueAsString(vo);
        System.out.println(jsonStr);
    }
 public JsonVO getAllVo(){
        JsonVO vo=new JsonVO();
        vo.setName("jzk");
        vo.setBirth(new Date());
        vo.setId(1l);
        vo.setCreatTime(new Date());
        vo.setMoney(true);
        vo.setWeight(new BigDecimal("20.56"));
        vo.setAddress(new Address("四川","成都"));
        List<Address> list=new ArrayList<Address>();
        list.add(new Address("四川","绵阳"));
        list.add(new Address("四川","简阳"));
        vo.setAddresses(list);
        Map<String,Object> son=new HashMap<>();
        son.put("name","将填好");
        son.put("sex","1");
        vo.setSon(son);
        vo.setWifes(new String[]{"冰冰","雅致"});
        vo.setBirth2(java.sql.Date.valueOf("2012-10-01"));
        return vo;
    }
效果
{
  "id" : 1,
  "name" : "jzk",
  "weight" : 20.56,
  "birth" : 1701917471582,
  "creatTime" : 1701917471582,
  "money" : true,
  "address" : {
    "shen" : "四川",
    "city" : "成都"
  },
  "addresses" : [ {
    "shen" : "四川",
    "city" : "绵阳"
  }, {
    "shen" : "四川",
    "city" : "简阳"
  } ],
  "son" : {
    "sex" : "1",
    "name" : "将填好"
  },
  "wifes" : [ "冰冰", "雅致" ],
  "birth2" : 1349020800000
}
4.4.5 把null全部转化成""
重写setNullValueSerializer()方法
@Test
    public void t4() throws JsonProcessingException {
        ObjectMapper mapper=new ObjectMapper();
        mapper.getSerializerProvider().setNullValueSerializer(
                new JsonSerializer<Object>() {
                    @Override
                    public void serialize(Object paramT,
                                          JsonGenerator paramJsonGenerator, SerializerProvider paramSerializerProvider) throws IOException, JsonProcessingException {
                        paramJsonGenerator.writeString("");
                    }
                });
        JsonVO vo=new JsonVO();
        vo.setId(2l);
        vo.setBirth(new Date());
        String jsonStr="";
        jsonStr=mapper.writerWithDefaultPrettyPrinter().writeValueAsString(vo);
        System.out.println(jsonStr);
    }
效果:
{
  "id" : 2,
  "name" : "dddddd",
  "weight" : "",
  "birth" : 1701917789597,
  "creatTime" : "",
  "money" : "",
  "address" : "",
  "addresses" : "",
  "son" : "",
  "wifes" : "",
  "birth2" : ""
}
4.4.6日期处理
日期处理有两种方式:
1.在对象属性上配置注解
@JsonFormat(timezone = “GMT+8”, pattern = “yyyy-MM-dd”)
@JsonFormat(timezone = “GMT+8”, pattern = “yyyy-MM-dd HH:mm:ss”)
2.jackson配置类上使用
mapper.setDateFormat(new SimpleDateFormat(“yyyy-MM-dd”));
因为注解大于配置,如果yyyy-MM-dd比较多,setDateFormat配置"yyyy-MM-dd",注解只使用带时分秒的。
省略掉配置日期的注解
代码
 vo
ublic class JsonVO {
    private Long id;
    private String name
    private BigDecimal weight;//年纪
    @JsonFormat(timezone = "GMT+8", pattern = "yyyy-MM-dd")
    private Date birth;
    @JsonFormat(timezone = "GMT+8", pattern = "yyyy-MM-dd HH:mm:ss")
    private Date creatTime;
解析
public void t5() throws JsonProcessingException {
    ObjectMapper mapper=new ObjectMapper();
       //在序列化时自定义时间日期格式
    mapper.setDateFormat(new SimpleDateFormat("yyyy-MM-dd"));
    JsonVO vo=new JsonVO();
    vo.setId(2l);
    vo.setBirth(new Date());
    vo.setBirth2(java.sql.Date.valueOf("2012-12-01"));
    vo.setCreatTime(new Date());
    String jsonStr="";
    jsonStr=mapper.writerWithDefaultPrettyPrinter().writeValueAsString(vo);
    System.out.println(jsonStr);
}
效果
{
  "id" : 2,
  "name" : "dddddd",
  "weight" : null,
  "birth" : "2023-12-07",
  "creatTime" : "2023-12-07 11:54:14",
  "money" : null,
  "address" : null,
  "addresses" : null,
  "son" : null,
  "wifes" : null,
  "birth2" : "2012-12-01"
}
4.4.7 list处理
默认支持list参数
@Test
    public void t6() throws JsonProcessingException {
        ObjectMapper mapper = new ObjectMapper();
        List<Address> list=new ArrayList<>();
        list.add(new Address("四川","成都"));
        list.add(new Address("四川","简阳"));
        list.add(new Address());
        String jsonInString = mapper.writerWithDefaultPrettyPrinter().writeValueAsString(list);
        System.out.println(jsonInString);
    }
效果
[ {
  "shen" : "四川",
  "city" : "成都"
}, {
  "shen" : "四川",
  "city" : "简阳"
}, {
  "shen" : null,
  "city" : null
} ]
4.4.8 map处理
默认支持map
@Test
    public void t7() throws JsonProcessingException {
        ObjectMapper mapper = new ObjectMapper();
        Map<String,Object> map=new HashMap<>();
        map.put("address","天府大道");
        map.put("name","张三丰");
        String jsonInString = mapper.writerWithDefaultPrettyPrinter().writeValueAsString(map);
        System.out.println(jsonInString);
    }
效果
{
  "address" : "天府大道",
  "name" : "张三丰"
}
4.3.9 json转vo
注意事项:
1.假如json是单引号 ALLOW_SINGLE_QUOTES=true
2.假如json 中存在但 Java 对象不存在的属性 FAIL_ON_UNKNOWN_PROPERTIES=false
否则前两者要报错
3.json转list和map,需要使用constructCollectionType、constructMapType构造器注意对应的class
@Test
    public void t8() throws IOException {
        ObjectMapper mapper = new ObjectMapper();
     //允许json用单引号,不是双引号
    mapper.configure(JsonParser.Feature.ALLOW_SINGLE_QUOTES, true);
    //在反序列化时忽略在 json 中存在但 Java 对象不存在的属性
    mapper.configure(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES, false);
    //1.vo部分有值 ok
    JsonVO vo=new JsonVO();
    String jsonStr="";
    jsonStr="{'id':2,'name':'蒋增奎'}";
    vo=mapper.readValue(jsonStr,JsonVO.class);
    System.out.println(vo);
    //2.vo全部有值 ok
    jsonStr="{\"id\":1,\"name\":\"jzk\",\"weight\":20.56,\"birth\":\"2023-12-07\",\"creatTime\":\"2023-12-07 13:44:17\",\"money\":true,\"address\":{\"shen\":\"四川\",\"city\":\"成都\"},\"addresses\":[{\"shen\":\"四川\",\"city\":\"绵阳\"},{\"shen\":\"四川\",\"city\":\"简阳\"}],\"son\":{\"sex\":\"1\",\"name\":\"将填好\"},\"wifes\":[\"冰冰\",\"雅致\"],\"birth2\":\"2023-12-07\"}";
    vo=mapper.readValue(jsonStr,JsonVO.class);
    System.out.println(vo);
    //3.json有,但vo里没有对应属性,要使用
    //   mapper.configure(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES, false);
    //引号允许
    //   mapper.configure(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES, false);
    jsonStr="{'id':2,'name':'蒋增奎','waihao':'带头大哥'}";
    vo=mapper.readValue(jsonStr,JsonVO.class);
    System.out.println(vo);
    //4.直接转化成list
    //必须获得一个CollectionType,注入vo.class
    CollectionType javaType = mapper.getTypeFactory().constructCollectionType(List.class, Address.class);
    jsonStr="[{\"shen\":\"四川\",\"city\":\"成都\"},{\"shen\":\"四川\",\"city\":\"简阳\"},{\"shen\":null,\"city\":null}]";
    List<Address> list=mapper.readValue(jsonStr,javaType);
    System.out.println("list start -----");
    for (Address a:list){
        System.out.println(a);
    }
    //4.直接转化成map
    //第二参数是 map 的 key 的类型,第三参数是 map 的 value 的类型
    MapType mapJavaType = mapper.getTypeFactory().constructMapType(HashMap.class, String.class, Address.class);
    jsonStr="{\"address2\":{\"shen\":\"四川\",\"city\":\"简阳\"},\"address1\":{\"shen\":\"四川\",\"city\":\"成都\"}}";
    Map<String,Address> map=mapper.readValue(jsonStr,mapJavaType);
    for(String key:map.keySet()){
        Address value = map.get(key);
        System.out.println("key="+key+" vlaue="+value);
    }
}
4.4解析总结




















