代码工具1. 自动根据json字符串生成javabean类
2015-10-15 11:07
1076 查看
你还在为文档里铺天盖地滴变量, 结构复杂的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地址
https://github.com/boredream/CodeUtils
欢迎star 和 folk,除了这里的json解析工具还有其他功能
后续还会一直提供不同新功能,大家有需求也可以提出,特别常用的都会找时间尽量添加
好了,工具使用到此结束,后面介绍下代码原理等
----------------------------------------------------------------------
原理
项目为一个Java Project
利用Gson工具将字符串树结构解析出来,然后利用File IO流将bean结构根据需求生成代码再写入到文件中
可见项目中源码,良心注释,几乎2行代码1行注释
这里也贴出下核心类JsonUtils的代码
----------------------------------------------------------------------
问题
对于数组数据,其中元素的类型只会取第一个数据进行解析,
如果json字符串中数据内第一个数据不全或者数据为空,即无法获取
对于此类无法获取数据值造成无法判断类型的情况,都默认设为了Object类型
----------------------------------------------------------------------
欢迎Github上star和follow, 多个高质量源码等待着你
也可以在eoeandroid论坛主页中进入我个人中心,查看我其他主题帖子,篇篇高质量,你值得信赖
写错了一个类型,然后各种崩溃
写错了一个变量名无法解析
...
代码工具来啦~
自动生成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地址
https://github.com/boredream/CodeUtils
欢迎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); } // 获 c0b9 取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论坛主页中进入我个人中心,查看我其他主题帖子,篇篇高质量,你值得信赖
相关文章推荐
- 在线格式化代码工具
- mybatis generator生成代码工具的使用
- 【码云周刊第 27 期】代码生成器:提升程序员的生产力
- 广师Android群分享之Android开发实用小工具,提升开发效率
- 广师Android群分享之代码工具集合
- Ubuntu搭建OpenGrok
- Javascript中call的使用
- js-base64加密
- 详解JS位置、宽高属性之一:offset系列
- js实现MD5加密
- 编写高质量JavaScript代码绳之以法(The Essentials of Writing High Quality JavaScript)翻译
- JS实现选中当前菜单后高亮显示的导航条效果
- javascript小知识点(类和继承)
- JavaScript instanceof 运算符深入剖析
- 数字数组的全排列(js)
- jsp 导出excel
- jsonp应用
- jsonp应用
- jsonp应用
- js实现:点击一个按钮,弹出一个div,并向其中传值,修改后,再传出