json 转 javaBean
前言:经常在网络上看见一些关于json自动转换成javaBean的jar包,项目组里的人也在用,稍稍研究了下,都是用的反射来做的。我细细想了下里面的逻辑,我觉得直接生成JavaBean代码岂不是更加直接,反射在移动端,怎么说都是消耗资源的。
Java Bean 类的结构
首先要梳理下,到底Bean类是一个什么样的文件,很简单的是有很多属性的class文件,属性有很多类别,但是从json的角度来看,类别又是有限的:基本类型和List。我们可以用python来模拟下:
# encoding: utf-8
'''
Created on 2014.12.24 @author: 13051041
''' class JavaClass(object):
'''
class templete
'''
def __init__(self, name):
'''
Constructor
'''
self.mName = name;
self.mIntegerList = []
self.mStrList = []
self.mBolList = []
self.mListList = []
self.mClassList = []
self.mInClassTypeList = []
self.mIsParcelable = True
self.mPackageName = "package com.codegenerator;" def addIntAttribute(self, str1):
'''
add String attribute
'''
self.mIntegerList.append(str1) def addUnicodeAttribute(self, str1):
'''
add String attribute
'''
self.mStrList.append(str1) def addBooleanAttribute(self, bol):
'''
add Boolean attribute
'''
self.mBolList.append(bol) def addListAttribute(self, list1):
'''
add List attribute
'''
self.mListList.append(list1) def addClassAttribute(self, cls):
'''
add Class attribute
'''
self.mClassList.append(cls) def addInClassType(self, inClass):
'''
add Inner Class attribute
'''
self.mInClassTypeList.append(inClass) def test(self):
print "-----this is JavaClass information start-----"
print "class's name is %s" % self.mName
print "class's integer is %s " % self.mIntegerList
print "class's String is %s " % self.mStrList
print "class's Boolean is %s " % self.mBolList
print "class's List is %s " % self.mListList
print "class's inner class is %s " % self.mClassList
print "class's inner class type is %s " % self.mInClassTypeList
print "-----this is JavaClass information end-----" classNum = 0
def autoGeneratorClassName():
global classNum
classNum = classNum + 1
return "class%d" % classNum def buildJavaClassNode(jsonObject, className):
javaclass = JavaClass(className)
keys = jsonObject.keys()
for key in keys:
t = type(jsonObject[key])
if t == int:
javaclass.addIntAttribute(key)
if t == unicode:
javaclass.addUnicodeAttribute(key)
if t == bool:
javaclass.addBooleanAttribute(key)
if t == dict:
newClassName = autoGeneratorClassName()
javaclass.addClassAttribute((newClassName, key))
javaclass.addInClassType(buildJavaClassNode(jsonObject[key], newClassName))
if t == list:
t2 = type(jsonObject[key][0])
if t2 == dict:
newClassName = autoGeneratorClassName()
javaclass.addListAttribute((newClassName, key))
javaclass.addInClassType(buildJavaClassNode(jsonObject[key][0], newClassName))
else:
javaclass.addListAttribute((t2, key)) return javaclass
生成java文件,递归才是王道
我们一个叫javaclass的python类了,用python读取json,直接生成实例。现在的任务是把这个实例,写到文件里面。想想class的嵌套关系,生成最好的方法就是递归:
# encoding: utf-8
'''
Created on 2014.12.26 @author: 13051041
''' Class_import = '''
%s import java.util.ArrayList;
import java.util.List;
import org.json.JSONArray;
import org.json.JSONObject;
import org.json.JSONException;
''' Class_import_parcelable = '''
import android.os.Parcel;
import android.os.Parcelable;
''' Class_HEAD = '''
public %s class %s {
''' Class_HEAD_parcelable = '''
public %s class %s implements Parcelable{
public %s() {
} public %s(Parcel in) {
readFromParcel(in);
}
''' Class_Method_END = "}\n" Attribute_TEMP = "public %s %s;\n" Method_Analyzer = '''
public void analyzerJson(JSONObject jsonObject) throws JSONException {
if(jsonObject == null){
return;
}
''' Method_TEMP = "public %s %s(%s)throws JSONException {\n"
Method_GET_INT = '''%s = GetJsonAttributeUtil.getInt(jsonObject, "%s");'''
Method_GET_String = '''%s = GetJsonAttributeUtil.getString(jsonObject,"%s");'''
Method_GET_Boolean = '''%s = GetJsonAttributeUtil.getBoolean(jsonObject,"%s");'''
Method_GET_CLASS = '''%s = new %s();
%s.analyzerJson(GetJsonAttributeUtil.getJSONObject(jsonObject,"%s"));'''
Method_GET_List = '''
%s= new ArrayList<%s>();
JSONArray array%s = GetJsonAttributeUtil.getJSONArray(jsonObject,"%s");
int %sLen = array%s.length();
for(int i = 0;i<%sLen;i++){
%s item = new %s();
item.analyzerJson(array%s.getJSONObject(i));
%s.add(item);
}
''' Method_Parcelable_Write = '''public void writeToParcel(Parcel dest, int flags) {\n '''
Mehtod_Parcelable_Value_Write = "dest.write%s(%s);\n"
Method_Parcelable_Read = '''public void readFromParcel(Parcel in) {\n '''
Mehtod_Parcelable_Value_Read = "%s= in.readString(); " IS_PARCELABLE = True
FILE_NAME = "Bean" def writeParcelabelMethod(f, jcls):
# writeToParcel method
f.write(Method_Parcelable_Write)
for attribute in jcls.mIntegerList:
f.write("dest.writeInt(%s);\n" % (attribute))
for attribute in jcls.mStrList:
f.write("dest.writeString(%s);\n" % (attribute))
for attribute in jcls.mBolList:
f.write("dest.writeByte((byte) (%s ? 1 : 0));\n" % (attribute)) for attribute in jcls.mListList:
f.write("dest.writeTypedList(%s);" % (attribute[1])) for attribute in jcls.mClassList:
f.write("dest.writeParcelable(%s, PARCELABLE_WRITE_RETURN_VALUE);" % attribute[1]) f.write(Class_Method_END) # readFromParcel method
f.write(Method_Parcelable_Read)
for attribute in jcls.mIntegerList:
f.write("%s= in.readInt();\n" % (attribute))
for attribute in jcls.mStrList:
f.write("%s= in.readString();\n" % (attribute))
for attribute in jcls.mBolList:
f.write("%s = in.readByte() != 0;\n" % (attribute)) for attribute in jcls.mListList:
f.write("%s = new ArrayList<%s>();" % (attribute[1], attribute[0]))
f.write("in.readTypedList(%s, %s.CREATOR);" % (attribute[1], attribute[0])) for attribute in jcls.mClassList:
f.write("%s = in.readParcelable(%s.class.getClassLoader());" % (attribute[1], attribute[0])) f.write(Class_Method_END)
# describeContents method
f.write("public int describeContents() {return 0;}") f.write('''
public static final Parcelable.Creator<%s> CREATOR =
new Parcelable.Creator<%s>() {
public %s createFromParcel(Parcel in) {
return new %s(in);
} public %s[] newArray(int size) {
return new %s[size];
}
};
''' % (jcls.mName, jcls.mName, jcls.mName, jcls.mName, jcls.mName, jcls.mName)) pass def writeCls2File(jcls, f):
f.seek(0, 2)
# write class header
if IS_PARCELABLE: # if need implement parcelable
if jcls.mName == FILE_NAME:
f.write(Class_HEAD_parcelable % ("", jcls.mName, jcls.mName, jcls.mName))
else:
f.write(Class_HEAD_parcelable % ("static", jcls.mName, jcls.mName, jcls.mName))
else:
f.write(Class_HEAD % ("", jcls.mName)) # write class's attribute
for attribute in jcls.mIntegerList:
f.write(Attribute_TEMP % ('int', attribute))
for attribute in jcls.mStrList:
f.write(Attribute_TEMP % ('String', attribute))
for attribute in jcls.mBolList:
f.write(Attribute_TEMP % ('Boolean', attribute))
for attribute in jcls.mListList:
f.write(Attribute_TEMP % ('List<' + attribute[0] + '>', attribute[1]))
for attribute in jcls.mClassList:
f.write(Attribute_TEMP % (attribute[0], attribute[1])) # write class's method analyzerJson
f.write(Method_Analyzer)
for attribute in jcls.mIntegerList:
f.write(Method_GET_INT % (attribute, attribute))
for attribute in jcls.mStrList:
f.write(Method_GET_String % (attribute, attribute))
for attribute in jcls.mBolList:
f.write(Method_GET_Boolean % (attribute, attribute))
for attribute in jcls.mClassList:
f.write(Method_GET_CLASS % (attribute[1], attribute[0], attribute[1], attribute[1]))
for attribute in jcls.mListList:
f.write(Method_GET_List % (attribute[1], attribute[0], attribute[1], attribute[1], attribute[1], attribute[1], attribute[1], attribute[0], attribute[0], attribute[1], attribute[1]))
f.write(Class_Method_END) # write parcelable method
if IS_PARCELABLE:
writeParcelabelMethod(f, jcls) # the method toString
f.write("public String toString(){")
f.write("String allInfo = ")
for attribute in jcls.mIntegerList:
f.write(''' " %s="+%s + ''' % (attribute, attribute))
for attribute in jcls.mStrList:
f.write(''' " %s="+%s + ''' % (attribute, attribute))
for attribute in jcls.mBolList:
f.write(''' " %s="+%s + ''' % (attribute, attribute)) for attribute in jcls.mListList:
f.write(''' " %s="+%s.toString() + ''' % (attribute[1], attribute[1])) for attribute in jcls.mClassList:
f.write(''' " %s="+%s.toString() + ''' % (attribute[1], attribute[1])) f.write('''" ";''')
f.write("return allInfo;")
f.write(Class_Method_END) # write class's class
for innerClass in jcls.mInClassTypeList:
writeCls2File(innerClass, f)
pass
f.write(Class_Method_END)
pass def generatorClsFile(jcls):
global IS_PARCELABLE
global FILE_NAME
IS_PARCELABLE = jcls.mIsParcelable
FILE_NAME = jcls.mName
f = open(FILE_NAME + ".java", 'w')
f.write(Class_import % jcls.mPackageName)
if IS_PARCELABLE:
f.write(Class_import_parcelable)
writeCls2File(jcls, f)
完整项目:http://files.cnblogs.com/files/chenjie0949/CodeGenerator1.3.zip
json 转 javaBean的更多相关文章
- json、javaBean、xml互转的几种工具介绍
json.javaBean.xml互转的几种工具介绍 转载至:http://blog.csdn.net/sdyy321/article/details/7024236 工作中经常要用到Json.Jav ...
- Json与javaBean之间的转换工具类
/** * Json与javaBean之间的转换工具类 * * {@code 现使用json-lib组件实现 * 需要 * json-lib-2.4-jdk15.jar * ...
- json、javaBean、xml互转的几种工具介绍 (转载)
工作中经常要用到Json.JavaBean.Xml之间的相互转换,用到了很多种方式,这里做下总结,以供参考. 现在主流的转换工具有json-lib.jackson.fastjson等,我为大家一一做简 ...
- 关于flexjson将json转为javabean的使用
关于flexjson将json转为javabean的使用 import java.sql.Timestamp; import java.util.Date; import flexjson.JSOND ...
- Json、JavaBean、String等互转
Json.JavaBean.String等互转 本文介绍简单的Json.JavaBean.String互换(下文JavaBean简称Object对象,这里不是很严谨) 转换关系如下: 其中String ...
- IDEA使用GsonFormat完成JSON和JavaBean之间的转换
原文地址:https://www.leshalv.net/posts/12625/ 前言: 之前处理接口传来的一堆数据,用jsonObject很难受,后面就用gosn来弄,配合这个工具体验很好. 转: ...
- JSON 转javabean 利器
别再对着json来手写javabean啦.这个工作完全不要脑子,而且耗时. 这里给大家提供三种方式: android studio版: 万能的插件:GsonFormat 如何安装? Preferenc ...
- JSON转javabean(pojo)利器
别再对着json来手写javabean啦.这个工作完全不要脑子,而且耗时. 这里给大家提供三种方式: android studio版: 万能的插件:GsonFormat 如何安装? Preferenc ...
- Json、JavaBean、Map、XML之间的互转
思路是JavaBean.Map.XML都可以用工具类很简单的转换为Json,进而实现互相转换 1.Map.XML与Json互转 mvn依赖 <dependency> <groupId ...
随机推荐
- NOIP2010关押罪犯 二分+二染色
这个题一上来 没有思路,后来想没有思路就二分吧 那么我们来二分 首先,大于当前的mid值的关系,不能出现在一个集合里 (即关系形成的图是一个二分图,判定二分图可以二染色) 如果不能形成二分图,那么说明 ...
- python模拟登录人人网
参考: http://www.cnblogs.com/txw1958/archive/2012/03/12/2392067.html http://www.cnblogs.com/chenyg32/a ...
- Mathtype(对齐设置)
选左对齐 可以编辑->插入符号->空格符
- CodeForces 361B Levko and Permutation
题意:有n个数,这些数的范围是[1,n],并且每个数都是不相同的.你需要构造一个排列,使得这个排列上的数与它所在位置的序号的最大公约数满足 > 1,并且这些数的个数恰好满足k个,输出这样的一个排 ...
- sap判断条件
EQ 等于= 等于NE 不 等于<> 不 等于>< 不 等于LT 小 于< 小于LE 小 于等于<= 小 于等于GT 大 于> 大于GE ...
- scala学习笔记(四)样本类与模式匹配
访问修饰符 格式:private[x]或protected[x],x指某个所属包.类或单例对象,表示被修饰的类(或方法.单例对象),在X域中公开,在x域范围内都可以访问: private[包名]:在该 ...
- 常用的windowd属性和对象
window.location.href="" 指向一个定向的url并提交数据过去 window.location.reload() 强制刷新当前 ...
- bzoj3926: [Zjoi2015]诸神眷顾的幻想乡 对[广义后缀自动机]的一些理解
先说一下对后缀自动机的理解,主要是对构造过程的理解. 构造中,我们已经得到了前L个字符的后缀自动机,现在我们要得到L+1个字符的后缀自动机,什么需要改变呢? 首先,子串$[0,L+1)$对应的状态不存 ...
- 遇到Class Not registered的COM异常怎么办
博客搬到了fresky.github.io - Dawei XU,请各位看官挪步.最新的一篇是:遇到Class Not registered的COM异常怎么办.
- snowflake算法(java版)
转自:http://www.cnblogs.com/haoxinyue/p/5208136.html 1. 数据库自增长序列或字段 最常见的方式.利用数据库,全数据库唯一. 优点: 1)简单,代码方 ...