groovy/java自实现json解析器(2)JsonObject

简介: <div class="markdown_views"><h1 id="底层数据结构实现">底层数据结构实现</h1><p>本对象的底层数据结构是一个Map(映射),我们用<code>def private jsonMap</code>将其定义为对象变量。我们在构造函数中对其进行初始化,它以键值对的形式存储数据,其中键必须为字符串,值可以为字符串、Boolean、Inte

底层数据结构实现

本对象的底层数据结构是一个Map(映射),我们用def private jsonMap将其定义为对象变量。我们在构造函数中对其进行初始化,它以键值对的形式存储数据,其中键必须为字符串,值可以为字符串、Boolean、Integer、JsonArray、JsonObject,从最后两个可存储对象,我们或多或少地已能看出JsonObject是如何达成普通json对象里的无限嵌套了。
下面是本对象的构造函数。

def JsonObject( jsonMap = null) {
        this.jsonMap = jsonMap == null ? [:] : jsonMa//初始化jsonMap
    }

在调用构造函数时,如果我们不传入jsonMap,则会使用默认参数null,这时,我们就以[:]对map进行初始化。

公共调用API

下面定义了一些对外提供的API接口工具函数:

/**
* 根据键获取内容
 * @param key
 * @return
 */
def get(key){
    return jsonMap.get(key)
}
/**
 * 以key为键,将value存入map中
 * @param key
 * @param value
 * @return
 */
def put(key,value){
    jsonMap.put(key, value)
}
/**
 * 根据健删除内容,返回被删除的内容(如果不存在则返回null)
 * @param key
 * @return
 */
def romove(key){
    return jsonMap.remove(key)
}
/**
 * 返回键集
 * @return
 */
def keySet(){
    return jsonMap.keySet()
}
/**
 * 返回键值对集
 * @return
 */
def entrySet(){
    return jsonMap.entrySet()
}
/**
 * 判断是否为空
 */
def isEmpty(){
    return jsonMap.isEmpty()
}
/**
 * 返回对象的大小
 */
def size(){
    return jsonMap.size()
}

如果使用过类似于net.json等库函数操作json字符串的,就会对上述的方法非常熟悉了。

核心算法实现

在核心实现中,首先定义了toJsonObject(object)方法,它能将普通json字符串、集合或数组和普通数据类型转化为我们的jsonObject对象,它的定义形如:

/**
 * 将object内容转换成JsonObject ,
 * object 必须是普通json字符串、集合或数组和普通数据类型
 * @param object
 * @return
 */
final static JsonObject toJsonObject(object) {
    JsonObject jsonObject = null
    println object
    if (object.getClass() == String.class) {// 如果是普通字符串,则格式化字符串格式输入
    some codes...
    }else{//条件判断是对象类型还是非字符串普通数据类型还是数组或集合类型,遍历对象属性添加
    the other codes...
    }// end of 对象类型外循环
    jsonObject  //返回值
}
  1. 首先我们把object是否为字符串拿出来单独判断,如果object.getClass() == String.class,object必须以”{“开头,且以”}“结尾。这种情况我们单独解析,而关于如何把json格式字符串解析成我们的JsonObject,这里使用了两种方法:

    1. 第一种方法的思路是参考了逆波兰算法的模式括号匹配
      /**
                   * 方法一:非递归实现json字符串格式转化
                   */
                  //先检查是否满足json格式要求
                  if(!ValidateJson.validate(object)){//不满足要求
                      return null//返回空
                  }
                  String jsonStr = ValidateJson.jsonStr//object
                  println jsonStr
                  // 先判断字符串是否以“{}“开头结尾
      //          if (!jsonStr.startsWith("{") || !jsonStr.endsWith("}")) {
      //              throw new IllegalArgumentException("字符串必须以“{}“开头结尾")
      //          }
                  def  lrMArray = JsonTool.posBrackerOfString(jsonStr,"[","]")//获取json字符串中所有中括号的位置
                  def  lrBArray = JsonTool.posBrackerOfString(jsonStr,"{","}")//获取json字符串中所有大括号的位置
                  println "中括号:${lrMArray}"
                  println "大括号:${ lrBArray}"
      
                  //开始根据括号来构建jsonObject
                  //主要实现原理,根据lrBArray的特定,先存放的括号必定是最里面的,但后面存放的括号不一定包括前一个括号里面的内容
                  //故这里再次模拟一个栈来一次保存每个括号转换成的jsonObject对象,然后再后续进行序号比较
                  //若后面括号的起始端序号大于前面括号的起始端信号,说明是包含关系,则在转换后面括号为对象时,通过序号相等将前面括号对象作为后面括号对象的一员,
                  //注意此时应删除已应用的对象,具体见fromJsonObjectPart函数
                  jsonObject  = JsonTool.getJsonByMergeArray(jsonStr,lrMArray, lrBArray)//这里当前区域函数结束的最后一行,会将函数结果作为返回值返回。即返回我们最终的jsonObject对象

    (关于JsonArray的相关解析请看后面系列:”groovy/java自实现json解析器(3)JsonArray”部分,但鉴于json字符串对象和数组的循环嵌套特性,两者的结合分析将贯穿整个类各个函数。)

    我们知道,json字符串对象的键必须为字符串,而值可为布尔值、数字、字符串、对象、数组等,像布尔值、数字、字符这些不会存在嵌套,我们无需过多考虑,可以直接根据json字符串格式,直接解析,但如果值为对象或数组,则值对应的字符串又会以”{“开头,“}”结尾或以”[“开头,”]”结尾,这个时候,我们就不能简单地直接解析,我们需要先用两个栈(分别存储中括号和大括号的索引位置)来存储字符串出现的左括号,然后不断扫描json字符串,当检测到右括号时,就将之前的左括号弹出栈,将匹配的括号对在字符串中的索引位置存储在一个整型数组中,这是posBrackerOfString()函数的实现思路,它返回了所搜寻括号的所有对应索引位置,根据数组索引从小到大,对应括号位置从里到外。当我们分别获得中括号,大括号对应索引的两个数组时,我们需要对它们进行合并(合并算法有点像数据结构中的归并排序),即调用getJsonByMergeArray(),合并的结果就是我们最终需要的jsonObject对象了。关于这是posBrackerOfString和即getJsonByMergeArray定义请查看后面系列内容。

    1. 除了使用模式括号匹配来把字符串解析成我们需要的jsonObject,我们还可以通过相对”朴素“的递归实现。
      /**
                   *  方法二:通过递归实现对json格式字符串的转化,同时进行了校验,算法效率较高
                   */
                  jsonObject = StringToJson.jsonStringToObjectOrArray(object)

    因为无论多复杂的json字符串,最后还是归结为值为字符串、数字、布尔值的形式。所以从递归的思想来分析,凡是遇到字符串、数字、布尔值,就直接解析,遇到对象/数组,就新建一个jsonObject/jsonArray,不断地逐层深入进解析,总会有穷尽的时候,而一旦穷尽地进入最内层(没有了对象和数组),就是的开始了,最终,一定能把我们的json字符串解析成一个jsonObject和jsonArray复杂嵌套的jsonObject(考虑字符串充分复杂的话),具体实现在讲解StringToJson类中分析。

  2. 现在让我们看看object是:对象类型还是非字符串普通数据类型还是数组或集合类型,这个时候我们调用来递归获取
    jsonObject = recusiveToJsonObject(object)
    下面是recusiveToJsonObject函数的定义,它的实现思想体现在注释中。
/**
     * 非字符串Object递归处理部分,涵盖对象类型、基本数据类型、Map类型的处理
     * @param object
     * @return
     */
private final static recusiveToJsonObject(object){
    if(object instanceof Map){//将fieldData的Map中的属性转移到新的Map中
        def jsonMap = [:]
        for(entry in object){
            jsonMap[entry.key] = entry.value
        }
        return new JsonObject(jsonMap)
    }else if(object instanceof Collection || object.getClass().isArray()){//如果是数组类型,默认用”jsonArray"作键保存一个jsonArray
        return JsonArray.recusiveToJsonArray(object)
    }
    if(JsonTool.checkIsBasicClass(object.getClass())){//如果是非字符串的基本数据类型,则直接返回
        return object
    }
    //object 为对象类型,则通过反射遍历对象构建jsonObject
    def jsonMap = [:]
    Field[] fields = object.getClass().getDeclaredFields()
    for(i in 0..fields.length - 1){
        if(fields[i].name.contains("\$") || fields[i].name == "metaClass"){//过滤groovy默认添加的元属性和其他特定属性
            continue
        }
        Method method = null
        //先获取相应的get方法
        try {
            method = object.getClass().getMethod(JsonTool.getGetMethodName(fields[i].getName()))
        } catch (NoSuchMethodException e) {
            continue //不是需要格式化为json的方法,故跳过
        }
        //从对象中获取对应的属性
        def firldData = method.invoke(object)
        //判断该属性是基本类型还是复杂类型
        if(JsonTool.checkIsBasicClass(fields[i].getType())){//是基本的数据类型
            jsonMap.put(fields[i].getName(),firldData);
        }else  if(fields[i].getType() == Date.class){//如果是日期类型
            //先默认转换特定格式,后期补充自定义
            SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd hh:mm:ss")
            jsonMap.put(fields[i].getName(),sdf.format(firldData))
        }else if(fields[i] instanceof Map){//Map类型,转换成jsonObject
            jsonMap[fields[i].getName()] = recusiveToJsonObject(firldData);//递归调用创建新的对象并放入jsonMap中
        }else if(fields[i].getType().isArray() || fields[i] instanceof Collection){//数组类型
            jsonMap[fields[i].getName()] =  JsonArray.toJsonArray(firldData)
        }else{//复杂对象类型,直接放入
            jsonMap[fields[i].getName()] =  recusiveToJsonObject(firldData)
        }
    }//end of 属性添加到jsonMap的循环
     new JsonObject(jsonMap)
}

以上我们完成了object类型到JsonObject的转化,但实际比较常用的一个应用,就是把JsonObject,自动组装成我们需要的类对象,这个时候,我们就需要用到java的反射技术了。可惜的是,groovy是一门动态语言,它的动态特性会为我们在用反射构造对象时带来许多麻烦,以下是一个粗糙的实现:

/**
 * 递归将jsonObject根据类参数转化为特定的类对象
 * @param clazz //要转换为对象的类
 * @param jsonObject 
 * @return
 */
final static toBean(Class<?> clazz,JsonObject jsonObject){
    Object instance = clazz.newInstance()//初始化一个要转化的类对象
    for(Map.Entry<String,Object> entry : jsonObject.entrySet()){
        //根据entry对应的key名获取类相应的属性(用来获取属性的类型)set方法
        Field field = null
        Method method = null
        try{
            field = clazz.getDeclaredField(entry.getKey());//获取属性
        }catch(NoSuchFieldException e){
            System.out.println("没有在类"+clazz+"中找到"+jsonObject+"中的json健名:"+entry.getKey() + "请检查该类是否明确定义")
            throw e
        }
        try {
            method = clazz.getMethod(JsonTool.getSetMethodName(entry.getKey()),field.getType());//获取方法
        } catch (NoSuchMethodException e) {
            System.out.println("没有与"+entry.getKey()+"对应的公用set方法")
            throw e
        }

        //判断该属性是jsonObject或是jsonArray或是普通数据类型
        if(entry.getValue().getClass() == JsonObject.class){
            method.invoke(instance, toBean(field.getType(), (JsonObject) entry.getValue()));//递归调用,将jsonObject转换为bean再注入
        }else if(entry.getValue().getClass() == JsonArray.class){
            method.invoke(instance, JsonArray.toArray(field.getType(),entry.getValue()))
        }else{//普通数据类型,则entry.getValue类型统一为String
            method.invoke(instance, JsonTool.getValueByType(field.getType(),String.valueOf(entry.getValue())));//调用方法为属性注入值
        }
    }
    instance//返回值
}

这个方法在groovy中用得并不完美,在实际使用中还会有或多或少的bug,同样的json功能函数,在Java中就显得严谨健壮多了。

/**
     * 递归将jsonObject根据类参数转化为特定的类对象
     * @param clazz //要转换为对象的类
     * @param jsonObject 
     * @return
     * @throws IllegalAccessException 
     * @throws InstantiationException 
     * @throws NoSuchFieldException 
     * @throws NoSuchMethodException 
     * @throws ParseException 
     * @throws InvocationTargetException 
     * @throws IllegalArgumentException 
     */
    public Object toBean(Class<?> clazz, JsonObject jsonObject) throws InstantiationException, IllegalAccessException, NoSuchFieldException, NoSuchMethodException, IllegalArgumentException, InvocationTargetException, ParseException{
Object instance = clazz.newInstance();//初始化一个要转化的类对象
Map<String,Object> jsonMap =jsonObject.getJsonMap();//获取map,并遍历中的所有属性
for(Map.Entry<String,Object> entry : jsonMap.entrySet()){
    //根据entry对应的key名获取类相应的属性(用来获取属性的类型)set方法
    Field field = null;
    Method method = null;
    try{
        field = clazz.getDeclaredField(entry.getKey());//获取属性
    }catch(NoSuchFieldException e){
        System.out.println("没有在类中找到json健名:"+entry.getKey());
        throw e;
    }
    try {
        method = clazz.getMethod(getSetMethodName(entry.getKey()),field.getType());//获取方法
    } catch (NoSuchMethodException e) {
        System.out.println("没有与"+entry.getKey()+"对应的公用set方法");
        throw e;
    }

    //判断该属性是jsonObject或是jsonArray或是普通数据类型
    if(entry.getValue().getClass() == JsonObject.class){
        method.invoke(instance, toBean(field.getType(), (JsonObject) entry.getValue()));//递归调用,将jsonObject转换为bean再注入
    }else if(entry.getValue().getClass() == JsonArray.class){
        method.invoke(instance, JsonArray.toArray(field.getType(),(JsonArray)entry.getValue()));
    }else{//普通数据类型,则entry.getValue类型统一为String
        method.invoke(instance, JsonTool.getValueByType(field.getType(),String.valueOf(entry.getValue())));//调用方法为属性注入值
    }
}

return instance;
}

toString方法重写

最后,也是每个对象常常具备且需重写的”toString”方法,但在我们这个对象中,我们不能单纯地把它的对象属性:jsonMap println出来,那不是我们想要的,我们应该把它解析成一个合格的json字符串来返回:

/**
 * 重写toString函数,将jsonObject转换成json字符串格式输出
 */
@Override
def String toString() {
    StringBuffer str = JsonTool.objectToString(this,new StringBuffer());
    String jsonStr = str.substring(0,str.length() - 1);
    return jsonStr;
}

它的具体实现将会在jsonTool类讲解中看到。

目录
相关文章
|
8月前
|
JSON 网络协议 安全
【Java】(10)进程与线程的关系、Tread类;讲解基本线程安全、网络编程内容;JSON序列化与反序列化
几乎所有的操作系统都支持进程的概念,进程是处于运行过程中的程序,并且具有一定的独立功能,进程是系统进行资源分配和调度的一个独立单位一般而言,进程包含如下三个特征。独立性动态性并发性。
416 1
|
8月前
|
JSON 网络协议 安全
【Java基础】(1)进程与线程的关系、Tread类;讲解基本线程安全、网络编程内容;JSON序列化与反序列化
几乎所有的操作系统都支持进程的概念,进程是处于运行过程中的程序,并且具有一定的独立功能,进程是系统进行资源分配和调度的一个独立单位一般而言,进程包含如下三个特征。独立性动态性并发性。
388 1
|
机器学习/深度学习 人工智能 Java
Java机器学习实战:基于DJL框架的手写数字识别全解析
在人工智能蓬勃发展的今天,Python凭借丰富的生态库(如TensorFlow、PyTorch)成为AI开发的首选语言。但Java作为企业级应用的基石,其在生产环境部署、性能优化和工程化方面的优势不容忽视。DJL(Deep Java Library)的出现完美填补了Java在深度学习领域的空白,它提供了一套统一的API,允许开发者无缝对接主流深度学习框架,将AI模型高效部署到Java生态中。本文将通过手写数字识别的完整流程,深入解析DJL框架的核心机制与应用实践。
963 3
|
存储 设计模式 Java
重学Java基础篇—ThreadLocal深度解析与最佳实践
ThreadLocal 是一种实现线程隔离的机制,为每个线程创建独立变量副本,适用于数据库连接管理、用户会话信息存储等场景。
534 5
|
存储 监控 安全
重学Java基础篇—类的生命周期深度解析
本文全面解析了Java类的生命周期,涵盖加载、验证、准备、解析、初始化、使用及卸载七个关键阶段。通过分阶段执行机制详解(如加载阶段的触发条件与技术实现),结合方法调用机制、内存回收保护等使用阶段特性,以及卸载条件和特殊场景处理,帮助开发者深入理解JVM运作原理。同时,文章探讨了性能优化建议、典型异常处理及新一代JVM特性(如元空间与模块化系统)。总结中强调安全优先、延迟加载与动态扩展的设计思想,并提供开发建议与进阶方向,助力解决性能调优、内存泄漏排查及框架设计等问题。
654 5
|
Java 开发者
重学Java基础篇—Java类加载顺序深度解析
本文全面解析Java类的生命周期与加载顺序,涵盖从加载到卸载的七个阶段,并深入探讨初始化阶段的执行规则。通过单类、继承体系的实例分析,明确静态与实例初始化的顺序。同时,列举六种触发初始化的场景及特殊场景处理(如接口初始化)。提供类加载完整流程图与记忆口诀,助于理解复杂初始化逻辑。此外,针对空指针异常等问题提出排查方案,并给出最佳实践建议,帮助开发者优化程序设计、定位BUG及理解框架机制。最后扩展讲解类加载器层次与双亲委派机制,为深入研究奠定基础。
547 0
|
安全 IDE Java
重学Java基础篇—Java Object类常用方法深度解析
Java中,Object类作为所有类的超类,提供了多个核心方法以支持对象的基本行为。其中,`toString()`用于对象的字符串表示,重写时应包含关键信息;`equals()`与`hashCode()`需成对重写,确保对象等价判断的一致性;`getClass()`用于运行时类型识别;`clone()`实现对象复制,需区分浅拷贝与深拷贝;`wait()/notify()`支持线程协作。此外,`finalize()`已过时,建议使用更安全的资源管理方式。合理运用这些方法,并遵循最佳实践,可提升代码质量与健壮性。
529 1
|
数据采集 JSON 数据可视化
JSON数据解析实战:从嵌套结构到结构化表格
在信息爆炸的时代,从杂乱数据中提取精准知识图谱是数据侦探的挑战。本文以Google Scholar为例,解析嵌套JSON数据,提取文献信息并转换为结构化表格,通过Graphviz制作技术关系图谱,揭示文献间的隐秘联系。代码涵盖代理IP、请求头设置、JSON解析及可视化,提供完整实战案例。
864 4
JSON数据解析实战:从嵌套结构到结构化表格
|
传感器 监控 Java
Java代码结构解析:类、方法、主函数(1分钟解剖室)
### Java代码结构简介 掌握Java代码结构如同拥有程序世界的建筑蓝图,类、方法和主函数构成“黄金三角”。类是独立的容器,承载成员变量和方法;方法实现特定功能,参数控制输入环境;主函数是程序入口。常见错误包括类名与文件名不匹配、忘记static修饰符和花括号未闭合。通过实战案例学习电商系统、游戏角色控制和物联网设备监控,理解类的作用、方法类型和主函数任务,避免典型错误,逐步提升编程能力。 **脑图速记法**:类如太空站,方法即舱段;main是发射台,static不能换;文件名对仗,括号要成双;参数是坐标,void不返航。
617 5
|
Java API 数据处理
深潜数据海洋:Java文件读写全面解析与实战指南
通过本文的详细解析与实战示例,您可以系统地掌握Java中各种文件读写操作,从基本的读写到高效的NIO操作,再到文件复制、移动和删除。希望这些内容能够帮助您在实际项目中处理文件数据,提高开发效率和代码质量。
938 4

推荐镜像

更多
  • DNS