基于动态字节码编译技术的codejson框架的制作方法_2

文档序号:9810288阅读:来源:国知局
列化性能较Fast json提高了 15% ; Jf ire_codejson框架提供了完全的反序列化能力,针对任何符合Json规范的字符串均均可以正确的反序列化;在性能强劲的同时还具备非常简单的API接口,对于常规操作,序列化和反序列化各自只有一个接口调用,非常简单易用,Jfire-code json框架还支持以插件形式进行功能扩展,非常方便进行自定义的开发。
【附图说明】
[0033]下面参照附图结合实施例对本发明作进一步的说明。
[0034]图1为本发明一种基于动态字节码编译技术的codejson框架中序列化方法的执行流程图。
[0035]图2为本发明一种基于动态字节码编译技术的codejson框架中反序列化方法的执行流程图。
【具体实施方式】
[0036]Jfire-codejson框架性能如此之强原因在于采用了与众不同的算法。
[0037]序列化:
[0038]传统的序列化框架,或者说相对比较优秀的序列化框架大部分都采用了分析对象内容,而后通过反射调用方法或者反射取出属性值的方式将对象进行序列化。这种思想产生的框架最大的瓶颈在反射导致的性能消耗。而Jfire-codejson独特的使用了为序列化对象动态编译一个输出对象,输出对象中全部是调用序列化对象的get方法来得到属性值。并且在拼接json中的key名称时,由于动态编译代码,所以编译代码中是事先知道并且写入的,这样又减少了获取对象属性名称的这一步骤,使得Jfire-codejson的序列化能力逼近理论上限(为每个对象编写代码的这种理论方式)。
[0039]如果要将一个对象序列化,最快的方式无疑是针对这个对象,写出一段特定的代码。代码当中通过调用对象的get方法来得到对象属性值。参考下面的代码:
[0040]1.public class User
[0041]2.{
[0042]3.private String name;
[0043]4.private int age;
[0044]5.}
[0045]6.//针对上面的类,最快的序列化json的代码应该是
[0046]7.public static void main(String args[])
[0047]8.{
[0048]9.User user = new UserO ;
[0049]10.StringBuilder str = new StringBuilder();
[0050]11.str.append("{");
[OO51 ] 12.str.append(〃\〃name\〃: 〃);//无需放入变量值,采用固定值节省了变量获取的时间还有利于jvm优化
[0052]13.str.append("\"").append(user.getName()).append("\",");
[0053]14.str.append("\"age\":").append(user.getAge());
[0054]15.str.append("}");
[0055]16.}
[0056]在上面的示例代码中,没有反射,没有任何分析。针对对象给出特定的序列化代码。如果Jf ire-code json框架也能够模拟这样的方式,就可以达到最大化的序列速度。并且可以逼近理论的上限。为了达到这样的效果,采用了动态代码编译的方式来完成。本发明的序列化方法是基于以上的思想来实现的。
[0057]本发明的一种基于动态字节码编译技术的code json框架,所述code json框架用于实现序列化方法与反序列方法;
[0058]具体地,如图1所述,所述序列化方法包括如下步骤:
[0059]步骤Al、获取目标对象,在WriteContext (^riteContext是一个最普通的Map结构,用来存储目标对象类和输出对象实例的映射关系)中寻找该目标对象特定的序列化器;[ΟΟ?Ο] 步骤Α2、若WriteContext中存在所述特定的序列化器,则直接使用所述特定的序列化器对该目标对象进行序列化后,结束流程;否则,通过动态字节码编译技术生成一个新的输出对象(输出对象与目标对象--对应,输出对象专门用来将制定的目标对象序列化为json字符串),并且该输出对象具有一个指定好的输出接口 ;
[0061]步骤A3、往该输出接口的方法体中添加字符缓存器,所述字符缓存器是作为字符串插入的容器;
[0062]步骤A4、分析所述目标对象的信息,抽取出该目标对象中所有符合Javabean规范的get方法,对每一所述get方法的返回值进行分析,该返回值的类型包括基本类型、包装类和Str ing、实现11 erator接口类、实现Map接口类、数组以及用户自定义的类;
[0063]若该返回值属于基本类型,则往所述方法体中写入相应的代码(类似builder.append(entity.getint))将该返回值序列化为字符串,并存储至所述字符缓存器;
[0064]若该返回值属于包装类和String,则往所述方法体中写入相应的代码(类似if(entity.geName()! =null) (builder.append (entity.geName())将该返回值序列化为字符串,并存储至字符缓存器;
[0065]若该返回值属于实现Iterator接口类,则往所述方法体中写入相应的代码判断该返回值是否为空,如果不为空,则再写入相应的代码遍历Iterator内容的代码,代码内部以调用WriteContext输出为字节码编译内容,即将该返回值序列化为字符串,并存储至字符缓存器,如果为空,则不进行操作;
[0066]若该返回值属于实现Map接口类,则往所述方法体中写入相应的代码判断该返回值是否为空,如果该返回值不为空,则再写入相应的代码用于遍历Map内容,并判断Map内容中每一对键值对的键是否为空,如果键为空,则不进行操作,如果键不为空,则再写入相应的代码将该返回值序列化为字符串,并存储至字符缓存器,如果该返回值为空,则不进行操作;
[0067]若该返回值属于数组,则往所述方法体中写入相应的代码判断该返回值是否为空,如果该返回值不为空,则再写入相应的代码用于遍历当前维度数组,如果当前数组维度大于或等于2,则判断当前数组是否为空,如果当前数组为空,则不进行操作,如果当前数组不为空,则继续遍历当前维度数组;如果当前数组维度小于2,则填充调用WriteContextWrite来输出内容的字节码,即将该返回值序列化为字符串,并存储至字符缓存器,如果该返回值为空,则不进行操作;
[0068]若该返回值属于用户自定义的类,则往所述方法体中写入相应的代码判断该返回值是否为空,如果不为空,则再写入相应的代码将该返回值序列化为字符串,并存储至字符缓存器,如果为空,则不进行操作;
[0069 ]步骤A5、分析完毕后,将所述字符缓存器中的字符串组装成特定的序列化输出类。[°07°] 通过上述的步骤,Jf ire-code json框架针对每一个对象,都能生成一个针对该对象的特定输出类,这个输出类为原生代码编译,获取对象内容以及属性名称都是原生代码调用,性能非常高。
[0071]当然,动态代码生成中有很多需要判断的地方。比如如果是对象的情况下,要在字符串中添加{};比如字符串需要使用〃包围内容,而数字和布尔值不需要;比如遇到数组需要[]等等。但是大体的思想就是通过动态代码编译,在运行的时候针对一个特定的对象,生成针对该对象的输出类。并且对这个类进行编译生成Class文件。动态编译这个部分采用Javassist来完成。生成的代码例子如下:
[0072]序列化一个如下对象:
[0073]1.public class com.jfire.codejson.Home
[0074]2.{
[0075]3.private String name = "home";
[0076]4.private int length = 113;
[0077]5.private int wdith = 89;
[0078]6.//省略get set方法
[0079]7.}
[0080]Jf ire-code json框架针对这个对象生成的动态类的内容是:
[0081]1.public class Jsonffriter_Home_231313131
[0082]2.{
[0083]3.StringCache cache = (StringCache)$2;
[0084]4.com.jfire.codejson.Home entity = (com.jfire.codejson.Home)$1;
[0085]5.cache.append('{');
[0086]6.cache.appenWlengthV:.append(entity.getLength()).append(',);
[0087]7.String name = entity.getName();
[0088]8.1f (name! =null)
[0089]9.{
[0090]10.cache.append("\"name\":\"").append(name).append("\",");
[0091]11.}
[0092]12.cache.append("\"wdith\":").append(entity.getffdith()).append(,,,);
[009
当前第2页1 2 3 4 
网友询问留言 已有0条留言
  • 还没有人留言评论。精彩留言会获得点赞!
1