代码工具1. 自动根据json字符串生成javabean类

本文介绍了如何使用代码工具自动根据json字符串生成JavaBean类,从而简化开发工作,提高效率。工具支持一键生成,避免手动编写带来的错误。文章还提供了使用方法、核心代码示例,并提醒了当json数据中存在无法判断类型的值时,会默认使用Object类型。此外,工具已开源并在GitHub上提供,鼓励用户star和fork,作者还会不断添加新功能以满足更多需求。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

你还在为文档里铺天盖地滴变量, 结构复杂的json而烦恼吗~
写错了一个类型,然后各种崩溃
写错了一个变量名无法解析
...

代码工具来啦~
自动生成javabean类,只要一键~

懒是一种程序员的必备美德,程序员都应该学会躲懒
撸的过程中一切重复枯燥无技术含量的工作都是可耻的,
此系列的代码工具就致力于此,替你完成这些无聊的工作,让你有更多的精力时间研究新姿势

----------------------------------------------------------------------

使用方法
先将代码工具导入Eclipse, 该项目为Java Project,注意不是Android Project

1.将json字符串数据复制到项目中的jsonString.txt文件中
(json数据可以从文档中的json数据示例里,或先运行一遍接口调用代码打印出来json字符串获取到)


2.运行代码工具中的JsonUtils类,其中的main函数中会运行json解析主方法


3.打开JsonBean.java文件,刷新,复制其中自动生成好的javabean类直接使用


补充
如果需要public修饰变量,可以ctrl+F进行全部替换
如果需要生成getter and setter方法,可以右键Source->Generate Getter and Setter进行设置


工具代码已开源,也可以根据需求自行修改

Github地址
欢迎star 和 folk,除了这里的json解析工具还有其他功能
后续还会一直提供不同新功能,大家有需求也可以提出,特别常用的都会找时间尽量添加


好了,工具使用到此结束,后面介绍下代码原理等

----------------------------------------------------------------------

原理
项目为一个Java Project
利用Gson工具将字符串树结构解析出来,然后利用File IO流将bean结构根据需求生成代码再写入到文件中

可见项目中源码,良心注释,几乎2行代码1行注释
这里也贴出下核心类JsonUtils的代码

package utils;

import java.io.File;
import java.util.ArrayList;
import java.util.Iterator;
import java.util.List;
import java.util.Map.Entry;

import com.google.gson.JsonArray;
import com.google.gson.JsonElement;
import com.google.gson.JsonObject;
import com.google.gson.JsonParser;
import com.google.gson.JsonPrimitive;

import entity.ArrayType;
import entity.Json2JavaElement;

public class JsonUtils {
    
     public static void main(String[] args) {
          parseJson2Java();
     }
    
     /**
     * 将json字符串转换为对应的javabean
     *
     * <p>
     * 用法:<br>
     * 将json字符串拷贝至本项目中/Json/JsonString.txt 文件中去,然后调用该方法,<br>
     * 就会在本项目中/Json/JsonBean.java中生成一个对应的JavaBean类<br><br>
     * 注意:<br>
     * 如果json字符串中有null或者空集合[]这种无法判断类型的,会统一使用Object类型
     */
     public static void parseJson2Java() {
          /// 读取json字符串
          String string = FileUtils.readToString(new File("Json\\JsonString.txt"), "UTF-8");
         
          // 解析获取整个json结构集合
          JsonParser parser = new JsonParser();
          JsonElement element = parser.parse(string);
          JsonObject jo = element.getAsJsonObject();
          List<Json2JavaElement> jsonBeanTree = getJsonBeanTree(jo);
         
          // 利用获取到的json结构集合,创建对应的javabean文件内容
          String javaBeanStr = createJavaBean(jsonBeanTree);
         
          // 将生成的内容写入到文件中去
          FileUtils.writeString2File(javaBeanStr, new File("Json\\JsonBean.java"));
     }
    
     /**
     * 根据解析好的数据创建生成对应的javabean类字符串
     *
     * @param jsonBeanTree 解析好的数据集合
     * @return 生成的javabean类字符串
     */
     private static String createJavaBean(List<Json2JavaElement> jsonBeanTree) {
          StringBuilder sb = new StringBuilder();
         
          // 是否包含自定义子类
          boolean hasCustomeClass = false;
          List<String> customClassNames = new ArrayList<String>();
         
          sb.append("public class JsonBeans {\n");
         
          // 由于在循环的时候有移除操作,所以使用迭代器遍历
          Iterator<Json2JavaElement> iterator = jsonBeanTree.iterator();
          while(iterator.hasNext()) {
               Json2JavaElement j2j = iterator.next();
              
               // 保存自定义类名称至集合中,注意已经包含的不再添加
               if(j2j.getCustomClassName() != null && !customClassNames.contains(j2j.getCustomClassName())) {
                    customClassNames.add(j2j.getCustomClassName());
               }
              
               if(j2j.getParentJb() != null) {
                    // 如果有parent,则为自定义子类,设置标识符不做其他操作
                    hasCustomeClass = true;
               } else {
                    // 如果不是自定义子类,则根据类型名和控件对象名生成变量申明语句
                    // private TextView tv_name;
                    sb.append("\tprivate ")
                         .append(getTypeName(j2j))
                         .append(" ")
                         .append(j2j.getName())
                         .append(";\n");
                   
                    // 已经使用的数据会移除,则集合中只会剩下自定义子类相关的元素数据,将在后续的循环中处理
                    iterator.remove();
               }
          }
         
          // 设置所有自定义类
          if(hasCustomeClass) {
               for(String customClassName : customClassNames) {
                    // 根据名称申明子类
                   
                    // /*sub class*/
                    // public class CustomClass {
                    sb.append("\n\t/*sub class*/\n");
                    sb.append("\tpublic class ")
                         .append(customClassName)
                         .append(" {\n");
                   
                    // 循环余下的集合
                    Iterator<Json2JavaElement> customIterator = jsonBeanTree.iterator();
                    while(customIterator.hasNext()) {
                         Json2JavaElement j2j = customIterator.next();
                        
                         // 根据当前数据的parent名称,首字母转为大写生成parent的类名
                         String parentClassName = StringUtils.firstToUpperCase(j2j.getParentJb().getName());
                        
                         // 如果当前数据属于本次外层循环需要处理的子类
                         if(parentClassName.equals(customClassName)) {
                              // 根据类型名和控件对象名生成变量申明语句
                              // private TextView tv_name;
                              sb.append("\t\tprivate ")
                                   .append(getTypeName(j2j))
                                   .append(" ")
                                   .append(j2j.getName())
                                   .append(";\n");
                             
                              // 已经使用的数据会移除,减少下一次外层循环的遍历次数
                              customIterator.remove();
                         }
                    }
                    sb.append("\t}\n");
               }
          }
         
          sb.append("}");
          return sb.toString();
     }
    
     /**
     *  递归遍历整个json数据结构,保存至jsonBeans集合中
     * 
     *  @param rootJo 根json对象
     *  @return 解析好的数据集合
     */
     private static List<Json2JavaElement> getJsonBeanTree(JsonObject rootJo) {
          jsonBeans = new ArrayList<Json2JavaElement>();
          recursionJson(rootJo, null);
          return jsonBeans;
     }
    
     /**
     * 保存递归获取到数据的集合
     */
     private static List<Json2JavaElement> jsonBeans = new ArrayList<Json2JavaElement>();
     /**
     * 递归获取json数据
     *
     * @param jo 当前递归解析的json对象
     * @param parent 已经解析好的上一级数据,无上一级时传入null
     */
     private static void recursionJson(JsonObject jo, Json2JavaElement parent) {
          // 循环整个json对象的键值对
          for (Entry<String, JsonElement> entry : jo.entrySet()) {
               // json对象的键值对建构为 {"key":value}
               // 其中,值可能是基础类型,也可能是集合或者对象,先解析为json元素
               String name = entry.getKey();
               JsonElement je = entry.getValue();
              
               Json2JavaElement j2j = new Json2JavaElement();
               j2j.setName(name);
               if(parent != null) {
                    j2j.setParentJb(parent);
               }
              
               // 获取json元素的类型,可能为多种情况,如下
               Class<?> type = getJsonType(je);
               if(type == null) {
                    // 自定义类型
                   
                    // json键值的首字母转为大写,作为自定义类名
                    j2j.setCustomClassName(StringUtils.firstToUpperCase(name));
                    // ?
                    j2j.setSouceJo(je.getAsJsonObject());
                    jsonBeans.add(j2j);
                   
                    // 自定义类需要继续递归,解析自定义类中的json结构
                    recursionJson(je.getAsJsonObject(), j2j);
               } else if(type.equals(JsonArray.class)) {
                    // 集合类型
                   
                    // 重置集合数据,并获取当前json元素的集合类型信息
                    deepLevel = 0;
                    arrayType = new ArrayType();
                    getJsonArrayType(je.getAsJsonArray());
                   
                    j2j.setArray(true);
                    j2j.setArrayDeep(deepLevel);
                   
                    if(arrayType.getJo() != null) {
                         j2j.setCustomClassName(StringUtils.firstToUpperCase(name));
                         // 集合内的末点元素类型为自定义类, 递归
                         recursionJson(arrayType.getJo(), j2j);
                    } else {
                         j2j.setType(arrayType.getType());
                    }
                    jsonBeans.add(j2j);
               } else {
                    // 其他情况,一般都是String,int等基础数据类型
                   
                    j2j.setType(type);
                    jsonBeans.add(j2j);
               }
          }
     }
    
     /**
     * 集合深度,如果是3则为ArrayList<ArrayList<ArrayList<>>>
     */
     private static int deepLevel = 0;
     /**
     * 集合类型数据,用于保存递归获取到的集合信息
     */
     private static ArrayType arrayType = new ArrayType();
     /**
     * 递归获取集合的深度和类型等信息
     *
     * @param jsonArray json集合数据
     */
     private static void getJsonArrayType(JsonArray jsonArray) {
          // 每次递归,集合深度+1
          deepLevel ++;
         
          if (jsonArray.size() == 0) {
               // 如果集合为空,则集合内元素类型无法判断,直接设为Object
               arrayType.setArrayDeep(deepLevel);
               arrayType.setType(Object.class);
          } else {
               // 如果集合非空则取出第一个元素进行判断
               JsonElement childJe = jsonArray.get(0);
              
               // 获取json元素的类型
               Class<?> type = getJsonType(childJe);
              
               if(type == null) {
                    // 自定义类型
                   
                    // 设置整个json对象,用于后续进行进一步解析处理
                    arrayType.setJo(childJe.getAsJsonObject());
                    arrayType.setArrayDeep(deepLevel);
               } else if (type.equals(JsonArray.class)) {
                    // 集合类型
                   
                    // 如果集合里面还是集合,则递归本方法
                    getJsonArrayType(childJe.getAsJsonArray());
               } else {
                    // 其他情况,一般都是String,int等基础数据类型
                   
                    arrayType.setArrayDeep(deepLevel);
                    arrayType.setType(type);
               }
          }
     }
    
     /**
     * 获取json元素的类型
     *
     * @param je json元素
     * @return 类型
     */
     private static Class<?> getJsonType(JsonElement je) {
          Class<?> clazz = null;
         
          if(je.isJsonNull()) {
               // 数据为null时,无法获取类型,则视为object类型
               clazz = Object.class;
          } else if(je.isJsonPrimitive()) {
               // primitive类型为基础数据类型,如String,int等
               clazz = getJsonPrimitiveType(je);
          } else if(je.isJsonObject()) {
               // 自定义类型参数则返回null,让json的解析递归进行进一步处理
               clazz = null;
          } else if(je.isJsonArray()) {
               // json集合类型
               clazz = JsonArray.class;
          }
          return clazz;
     }

     /**
     * 将json元素中的json基础类型,转换为String.class,int.class等具体的类型
     *
     * @param je json元素
     * @return 具体数据类型,无法预估的类型统一视为Object.class类型
     */
     private static Class<?> getJsonPrimitiveType(JsonElement je) {
          Class<?> clazz = Object.class;
          JsonPrimitive jp = je.getAsJsonPrimitive();
          // json中的类型会将数字集合成一个总的number类型,需要分别判断
          if(jp.isNumber()) {
               String num = jp.getAsString();
               if(num.contains(".")) {
                    // 如果包含"."则为小数,先尝试解析成float,如果失败则视为double
                    try {
                         Float.parseFloat(num);
                         clazz = float.class;
                    } catch(NumberFormatException e) {
                         clazz = double.class;
                    }
               } else {
                    // 如果不包含"."则为整数,先尝试解析成int,如果失败则视为long
                    try {
                         Integer.parseInt(num);
                         clazz = int.class;
                    } catch(NumberFormatException e) {
                         clazz = long.class;
                    }
               }
          } else if(jp.isBoolean()) {
               clazz = boolean.class;
          } else if(jp.isString()) {
               clazz = String.class;
          }
          // json中没有其他具体类型如byte等
          return clazz;
     }
    
     /**
     * 获取类型名称字符串
     *
     * @param j2j 转换数据元素
     * @return 类型名称,无法获取时,默认Object
     */
     private static String getTypeName(Json2JavaElement j2j) {
          String name = "Object";
         
          Class<?> type = j2j.getType();
          if(j2j.getCustomClassName() != null && j2j.getCustomClassName().length() > 0) {
               // 自定义类,直接用自定义的名称customClassName
               name = j2j.getCustomClassName();
          } else {
               // 非自定义类即可以获取类型,解析类型class的名称,如String.class就对应String
               name = type.getName();
               int lastIndexOf = name.lastIndexOf(".");
               if(lastIndexOf != -1) {
                    name = name.substring(lastIndexOf + 1);
               }
          }

          // 如果集合深度大于0,则为集合数据,根据深度进行ArrayList嵌套
          // 深度为3就是ArrayList<ArrayList<ArrayList<type>>>
          StringBuilder sb = new StringBuilder();
          for(int i=0; i<j2j.getArrayDeep(); i++) {
               sb.append("ArrayList<");
          }
          sb.append(name);
          for(int i=0; i<j2j.getArrayDeep(); i++) {
               sb.append(">");
          }
          return sb.toString();
     }
}



----------------------------------------------------------------------

问题
对于数组数据,其中元素的类型只会取第一个数据进行解析,
如果json字符串中数据内第一个数据不全或者数据为空,即无法获取
对于此类无法获取数据值造成无法判断类型的情况,都默认设为了Object类型

----------------------------------------------------------------------

欢迎Github上star和follow, 多个高质量源码等待着你
也可以在eoeandroid论坛主页中进入我个人中心,查看我其他主题帖子,篇篇高质量,你值得信赖




### 自动生成 Java Bean 为了实现根据 JSON 字符串自动生成 Java Bean 的功能,通常会涉及到解析 JSON 结构并将其映射为相应的 Java 属性和数据型。下面是一个基于给定的 JSON 字符串创建相应 Java Bean 的方法。 #### 解析 JSON生成结构 假设有一个 JSON 字符串如下: ```json { "id": 1, "username": "zs", "password": "123456", "email": "zs@163.com", "phone": "1386789898" } ``` 可以通过编写工具来分析此 JSON 对象中的键值对,并据此构建对应的 Java 定义。具体来说,对于上述例子应该生成这样的 `User` [^4]: ```java public class User { private Integer id; private String username; private String password; private String email; private String phone; // Getters and Setters... @Override public String toString() { return "User{" + "id=" + id + ", username='" + username + '\'' + ", password='" + password + '\'' + ", email='" + email + '\'' + ", phone='" + phone + '\'' + '}'; } } ``` #### 使用第三方库简化过程 实际开发中往往不会手动编码整个流程,而是借助像 Jackson 或 Gson 这样的序列化/反序列化框架来进行操作。这些库不仅能够方便地完成从 JSON 到对象以及反之的操作,还提供了辅助功能帮助开发者更轻松地处理复杂的嵌套结构等问题。 例如,在使用 FastJSON 库的情况下,可以直接利用其内置的方法快速将 JSON 转换成目标型的实例: ```java import com.alibaba.fastjson.JSON; // 把json转成JavaBean(user对象) String jsonStr = "{\"id\":1,\"username\":\"zs\",\"password\":\"123456\",\"email\":\"zs@163.com\",\"phone\":\"1386789898\"}"; User user = JSON.parseObject(jsonStr, User.class); System.out.println(user); // 同样也可以转换成Map形式存储 Map<String, Object> map = JSON.parseObject(jsonStr, Map.class); System.out.println(map); ``` #### 自动生成机制 如果希望进一步自动化这一过程,则可能需要考虑集成 IDE 插件或是命令行工具。比如可以在 IntelliJ IDEA 中配置动作以便于一键生成所需的 Java 文件[^2]: ```xml <action id="GenerateJavaBeanBySting" class="actions.generateJavaBean.GenerateJavaBeanAction" text="New Java Bean File By String" description="Generate JavaBean File By String" icon="/icons/ic_logo.png"> <add-to-group group-id="NewGroup" anchor="first"/> </action> ``` 这种设置允许用户通过菜单选项触发特定逻辑,从而依据输入的 JSON 数据动态创建新的 Java 源码文件。
评论 4
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值