Java遍历Json的简单实例
作者:Xiao白码
Java遍历Json实例
记录我在工作中遇到的问题以及我的解决方案,问题是:如何前端给后端的返回值中取出body内的每一条目中的key和value?遇到这个问题我还是毫无头绪的,虽然听说过一些非常好用的JSON工具,但是还是有点无从下手。
无从下手其实不可怕,可怕的是没有思路。
1.取出返回值里的body部分
JSONObject jsonObject = JSONObject.parseObject (res).getJSONObject("body");
这里用到的是阿里巴巴的fastjson工具,引用的类是com.alibaba.fastjson.JSONObject;,会返回一个只含有body的Json对象。
2.取出每一条之中的Key和Value
//遍历key和value for (Map.Entry<String, Object> entry : jsonObject.entrySet()) { System.out.println("这条JSON的Key是:"+entry.getKey()+"这条JSON的Value是:"+ entry.getValue()); }
上面的代码会输出jsonObject每一条Key和Value,这个循环我的大致理解就是用Map遍历Json.entryset()的输出对象,Json.entryset()输出的应该是一个set。
这样得到输出每一条数据的Key和Json。
遍历处理JsonObject的内容
目标
遍历jsonObject中value,value为已实例化的实体类;并载入该实体类获取其中属性和值,拼接成字符串。
知识点
1.Class类
Java程序在执行时,Java执行时系统一直对全部的对象进行所谓的执行时类型标识。
这项信息纪录了每一个对象所属的类。虚拟机通常使用执行时类型信息选准正确方法去执行,用来保存这些类型信息的类是Class类。
Class类封装一个对象和接口执行时的状态,当装载类时。Class类型的对象自己主动创建。Class 没有公共构造方法。
Class 对象是在载入类时由Java 虚拟机以及通过调用类载入器中的 defineClass 方法自己主动构造的,因此不能显式地声明一个Class对象。
虚拟机为每种类型管理一个独一无二的Class对象。也就是说,每一个类(型)都有一个Class对象。执行程序时。Java虚拟机(JVM)首先检查是否所要载入的类相应的Class对象是否已经载入。
假设没有载入,JVM就会依据类名查找.class文件,并将其Class对象载入。
主要的 Java 类型(boolean、byte、char、short、int、long、float 和 double)和keyword void 也都相应一个 Class 对象。
每一个数组属于被映射为 Class 对象的一个类,全部具有同样元素类型和维数的数组都共享该 Class 对象。
一般某个类的Class对象被加载内存,它就用来创建这个类的全部对象。
一、怎样得到Class的对象呢?有三种方法能够的获取:
1、调用Object类的getClass()方法来得到Class对象,这也是最常见的产生Class对象的方法。
比如:
MyObject x; Class c1 = x.getClass();
2、使用Class类的中静态forName()方法获得与字符串相应的Class对象。
比如:
Class c2=Class.forName(“MyObject”)
Employee必须是接口或者类的名字。
3、获取Class类型对象的第三个方法很easy。假设T是一个Java类型。那么T.class就代表了匹配的类对象。
比如
Class cl1 = Manager.class; Class cl2 = int.class; Class cl3 = Double[].class;
注意:Class对象实际上描写叙述的仅仅是类型。而这类型未必是类或者接口。
比如上面的int.class是一个Class类型的对象。
因为历史原因。数组类型的getName方法会返回奇怪的名字。
二、Class类的经常用法
getName()
一个Class对象描写叙述了一个特定类的属性,Class类中最经常使用的方法getName以 String 的形式返回此 Class 对象所表示的实体(类、接口、数组类、基本类型或 void)名称。newInstance()
Class另一个实用的方法能够为类创建一个实例,这种方法叫做newInstance()。比如:x.getClass.newInstance(),创建了一个同x一样类型的新实例。newInstance()方法调用默认构造器(无參数构造器)初始化新建对象。getClassLoader()
返回该类的类载入器。getComponentType()
返回表示数组组件类型的 Class。getSuperclass()
返回表示此 Class 所表示的实体(类、接口、基本类型或 void)的超类的 Class。isArray()
判定此 Class 对象是否表示一个数组类。
三、Class的一些使用技巧
1. forName和newInstance结合起来使用,能够依据存储在字符串中的类名创建对象。
比如:
Object obj = Class.forName(s).newInstance();
虚拟机为每种类型管理一个独一无二的Class对象。因此能够使用==操作符来比較类对象。
比如:
if(e.getClass() == Employee.class)…
2. Class类的 forName、getClass、getName方法
Class.forName()
:返回与给定的字符串名称相关联类或接口的Class对象。Class.forName("");
的作用是要求JVM查找并加载指定的类,假设在类中有静态初始化器的话,JVM必定会运行该类的静态代码段。入参为:String className
getClass方法:
- 类型:public final Class<? extends Object> getClass()
- 功能:返回该对象的运行时类的java.lang.Class对象(API上的解释)
- 由方法类型可以知道,该方法只能由类的实例变量调用;
- 当你要获得一个类的Class对象时(作函数参数的时候),你不能调用getClass方法,那你只能用类名.class来达到效果
getName方法:
- 类型:public String getName()
- 功能:以String形式返回次Class对象所表示的实体名称
JButton b1 = new JButton("button1"); System.out.println(b1.getClass());
输出:class javax.swing.JButton
JButton b1 = new JButton("button1"); System.out.println(b1.getName());
输出:javax.swing.JButton可以发现用class属性和getClass返回的输出是一样的,用getName返回的比前面两种少了class和一个空格。
3.遍历实体类的属性
//java中遍历实体类,获取属性名和属性值 public static void testReflect(Object model) throws Exception{ for (Field field : model.getClass().getDeclaredFields()) { field.setAccessible(true); System.out.println(field.getName() + ":" + field.get(model) ); } }
4. Java创建对象实例的三种方法
1.new
通常都是使用java的关键字new来创建对象实例。
若有一个Something类,则可以通过下面的语句创建Something类的对象实例并指定到变量obj。
Something somethingNew = new Something();
通过new创建对象实例必须把类名写在源代码里面。
2.clone
若程序写成如下,则可以根据当前对象(this)建立一个新实例对象(没有调用构造函数).
public class Something implements Cloneable{ private Something obj; public Something cloneSomething() { try { obj = (Something)this.clone(); // obj = (Something)clone(); } catch (CloneNotSupportedException e) { e.printStackTrace(); } return obj; } }
如果需要复制上面的那个obj指向的对象实例时,调用somethingNew.cloneSomething()方法就ok了。
但是为什么不直接使用somethingNew.clone()呢?
JDK中Object# clone()方法的原型是:
protected native Object clone() throws CloneNotSupportedException;
方法修饰符是protected,而不是public。
这种访问的不可见性使得我们对Object#clone()方法不可见。
所以,必需重写Object的clone方法后才能使用。
public Object clone()throws CloneNotSupportedException { Something obj; obj= (Something)super.clone(); return obj; }
值得注意的是 :
如果需要使用clone方法,必需实现java.lang.Cloneable接口,否则会抛出java.lang.CloneNotSupportedException。
另外clone方法所做的的操作是直接复制字段的内容,换句话说,这个操作并不管该字段对应的对象实例内容。
像这样字段对字段的拷贝(field to field copy)就成为"浅拷贝",clone方法所做的正是"浅拷贝".
3.newInstance
利用java.lang.Class类的newInstance方法,则可根据Class对象的实例,建立该Class所表示的类的对象实例。
创建Something类的对象实例可以使用下面的语句(这样需要一个已经存在的对象实例)。
somethingNew.getClass().newInstance(). //或者使用下面的语句(只需要存在相应的.class文件即可) Something instance = (Something) Class.forName("cn.softkid.test.Something").newInstance(); //如果包下不存在相应.class文件,则会抛出ClassNotFoundException。
注意 :
newInstance创建对象实例的时候会调用无参的构造函数,所以必需确保类中有无参数的构造函数,否则将会抛出java.lang.InstantiationException异常。无法进行实例化。
遍历JsonObject 的内容
/** * 解析制表的getDataSource方法,获取报表数据 * @param reportId * @param branchNo * @return */ @Override public String exportTXTFile(String reportId, String branchNo, String coopBranchNo, String periodDate) { Map<String, String> map = new HashMap<>(); // 根据报表xml获取templateType String templateType = getTemplateTypeFromXml(reportId); map.put(FieldValues.REPORT_ID, reportId); map.put(FieldValues.BRANCH_NO, branchNo); map.put(FieldValues.COOP_BRANCH_NO, coopBranchNo); map.put(FieldValues.PERIOD_DATE, periodDate); map.put(FieldValues.TEMPLATE_TYPE, templateType); //通过制表平台获取报表内容 JSONObject jsonObject = uraReportbuilderService.getDataSource(map); Map<String, Object> mapContent = jsonObject; StringBuffer text = new StringBuffer(); for (Object keyValue : mapContent.values()) { //keyValue中为ArrayList 代表该报表的DataSource(即数据库表)中的不同行;每一行都是一个实例化后的实体类 if (keyValue instanceof ArrayList) { for (int i=0; i< ((ArrayList) keyValue).size(); i++) { try{ Class ownClass = Class.forName(((ArrayList) keyValue).get(i).getClass().getName()); //这里需要用getClass().getName() Field[] fields = ownClass.getDeclaredFields(); for (Field field : fields) { field.setAccessible(true); text.append(field.get(((ArrayList)keyValue).get(i))); text.append("|"); } //换行 text.append("\r\n"); } catch (Exception e){ logger.error("读取报表数据出错",e.toString()); return "fail"; } } } } return text.toString(); }
总结
以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。