`

json与hibernate死循环解决办法

阅读更多
【问题】如题所示,在我们使用hibernate框架而又需要将对象转化为json的时候,如果配置了双向的关联关系,就会出现这个死循环问题
异常信息:
[java] view plaincopy在CODE上查看代码片派生到我的代码片
Method public java.lang.String org.apache.commons.lang.exception.NestableRuntimeException.getMessage(int) threw an exception when invoked on net.sf.json.JSONException: There is a cycle in the hierarchy! 
The problematic instruction: 
---------- 
==> ${msgs[0][0]} [on line 76, column 25 in org/apache/struts2/dispatcher/error.ftl] 
---------- 
 
Java backtrace for programmers: 
---------- 
freemarker.template.TemplateModelException: Method public java.lang.String org.apache.commons.lang.exception.NestableRuntimeException.getMessage(int) threw an exception when invoked on net.sf.json.JSONException: There is a cycle in the hierarchy! 
    at freemarker.ext.beans.SimpleMethodModel.exec(SimpleMethodModel.java:130) 
    at freemarker.ext.beans.SimpleMethodModel.get(SimpleMethodModel.java:138) 
    at freemarker.core.DynamicKeyName.dealWithNumericalKey(DynamicKeyName.java:111) 
    at freemarker.core.DynamicKeyName._getAsTemplateModel(DynamicKeyName.java:90) 
    at freemarker.core.Expression.getAsTemplateModel(Expression.java:89) 
    at freemarker.core.Expression.getStringValue(Expression.java:93) 
    at freemarker.core.DollarVariable.accept(DollarVariable.java:76) 
    at freemarker.core.Environment.visit(Environment.java:209) 
    at freemarker.core.MixedContent.accept(MixedContent.java:92) 
    at freemarker.core.Environment.visit(Environment.java:209) 
    at freemarker.core.IfBlock.accept(IfBlock.java:82) 
    at freemarker.core.Environment.visit(Environment.java:209) 
    at freemarker.core.IfBlock.accept(IfBlock.java:82) 
    at freemarker.core.Environment.visit(Environment.java:209) 
    at freemarker.core.MixedContent.accept(MixedContent.java:92) 
    at freemarker.core.Environment.visit(Environment.java:209) 
    at freemarker.core.Environment.process(Environment.java:189) 
    at freemarker.template.Template.process(Template.java:237) 
    at org.apache.struts2.dispatcher.Dispatcher.sendError(Dispatcher.java:748) 
    at org.apache.struts2.dispatcher.Dispatcher.serviceAction(Dispatcher.java:505) 
    at org.apache.struts2.dispatcher.ng.ExecuteOperations.executeAction(ExecuteOperations.java:77) 
    at org.apache.struts2.dispatcher.ng.filter.StrutsPrepareAndExecuteFilter.doFilter(StrutsPrepareAndExecuteFilter.java:91) 
    at org.apache.catalina.core.ApplicationFilterChain.internalDoFilter(ApplicationFilterChain.java:235) 
    at org.apache.catalina.core.ApplicationFilterChain.doFilter(ApplicationFilterChain.java:206) 
    at org.apache.catalina.core.StandardWrapperValve.invoke(StandardWrapperValve.java:233) 
    at org.apache.catalina.core.StandardContextValve.invoke(StandardContextValve.java:191) 
    at org.apache.catalina.core.StandardHostValve.invoke(StandardHostValve.java:127) 
    at org.apache.catalina.valves.ErrorReportValve.invoke(ErrorReportValve.java:103) 
    at org.apache.catalina.core.StandardEngineValve.invoke(StandardEngineValve.java:109) 
    at org.apache.catalina.connector.CoyoteAdapter.service(CoyoteAdapter.java:293) 
    at org.apache.coyote.http11.Http11Processor.process(Http11Processor.java:861) 
    at org.apache.coyote.http11.Http11Protocol$Http11ConnectionHandler.process(Http11Protocol.java:606) 
    at org.apache.tomcat.util.net.JIoEndpoint$Worker.run(JIoEndpoint.java:489) 
    at java.lang.Thread.run(Thread.java:619) 
Caused by: java.lang.NullPointerException 
    at freemarker.ext.beans.SimpleMemberModel.unwrapArguments(SimpleMemberModel.java:85) 
    at freemarker.ext.beans.SimpleMethodModel.exec(SimpleMethodModel.java:106) 
    ... 33 more 
关键字是net.sf.json.JSONException: There is a cycle in the hierarchy!,意思是在层次关系里有一个循环
【原因】为什么会这样呢?原因在于你要转化的对象里配置了对另外一个对象的关联,而那个对象里又配置了对你这个对象的关联。比如我的两个类叫做Shop(商店)和Staff(员工),一个商店可以有多个员工,所以我给这两个对象配置了双向的一对多和多对一的关联关系。这时候问题就出现了,JSON lib在把shop对象转化为json字符串的时候,发现shop里有个Set<Staff>,它就会去级联的把Set<Staff>转化为json字符串,在它遍历Set的时候,发现Staff里又有一个Shop对象,这时候它又会去尝试把shop转化为json字符串,然后就发现shop里又有Set<Staff>,如此周而复始,就形成了死循环。
【解决】问题清楚了,如何解决呢?我百度了一下,也找到了数十条的资料,但大都只是都说明了要用jsonConfig.setJsonPropertyFilter(new PropertyFilter(){}),而对于其中的参数和if语句该如何写,并没有一个说的很明白的。为此我查看了json-lib的源码,并进行了尝试,总结如下:
1,思路是在JSONObject把Shop对象转化为json字符串的时候,在中间加一道过滤,如果当前要转化的属性是Set并且属性名是staffs,那么就进行过滤
2,代码如下:
[java] view plaincopy在CODE上查看代码片派生到我的代码片
Map<String, Object> map=new HashMap<String, Object>(); 
map.put("shops", list); 
map.put("total", total); 
         
JsonConfig jsonConfig = new JsonConfig(); 
jsonConfig.setJsonPropertyFilter(new PropertyFilter() { 
    public boolean apply(Object obj, String name, Object value) { 
    if(value instanceof Set<?>&&name.equals("staffs")){ 
        return true; 
    }else{ 
        return false; 
    } 
   } 
}); 
         
return JSONObject.fromObject(map,jsonConfig); 
如上,PropertyFilter是json-lib提供的进行属性过滤的一个接口,具体的实现是由apply方法做的,那么我们就需要重写apply方法。
       此方法有三个参数,第一个是Object类型的,是你要转化的对象的类型(Shop);
                                           第二个参数是String类型,是你要过滤的属性的名称;
                                           第三个参数是Object类型的,是你要过滤的属性的值(值可能是String或其它类型的,所以用Object)。
                      返回值是boolean类型的,返回true,会进行过滤;返回false,不会进行过滤。
      if语句的写法就要根据实际的需要了,比如说我这里要解决死循环,就要实现把Shop里的Set<Staff> staffs属性过滤掉,那我的if语句就应该如上面那样写。总而言之就是JSON-lib在转化的时候,会对每个属性都调用这个apply方法,这样我们就要根据实际的业务需要,如果当前属性符合你的if条件,那你就要返回true,进行过滤。是使用||还是&&也要根据实际而定。
      这样配置后,再测试,就发现获取Shop的时候死循环问题已经不再出现了。
3,同理,Staff端也应该进行类似的配置
[java] view plaincopy在CODE上查看代码片派生到我的代码片
Map<String,Object> map=new HashMap<String, Object>(); 
map.put("staffs", list); 
map.put("total", total); 
         
JsonConfig jsonConfig = new JsonConfig();   
jsonConfig.setExcludes(new String[]{"handler","hibernateLazyInitializer"}); 
jsonConfig.setJsonPropertyFilter(new PropertyFilter() { 
       public boolean apply(Object obj, String name, Object value) { 
        if(name.equals("shop")){ 
            return true; 
        }else{ 
            return false; 
        } 
    } 
}); 
         
return JSONObject.fromObject(map,jsonConfig); 

经过测试,也是没有问题的。
这里大家也可以看见jsonConfig.setExcludes(new String[]{"handler","hibernateLazyInitializer"});,这一行是为了防止hibernate延迟加载造成的异常而设置的。
4,到这里大功告成了吗?不,我在测试的时候发现了一个更严重的问题,如果按照上面做这样配置,那我获取shop的时候,生成的json字符串里staffs的Set不见了;获取Staff的时候,它的属性shop在json字符串里也不见了!稍加分析就可以知道这是上面配置造成的。按上面的配置,Shop里的Set<Staff>被过滤掉了,“过滤掉”的含义不是不级联的转化Staff里的Shop了,而是直接连Set<Staff>都不转化了。这可坏了,我配置双向关联关系就是为了关联显示,你把我的属性过滤掉了,那我还配置双向关联干嘛?我还这么大费周章的来解决死循环干嘛?
      那么这个问题该如何解决呢?其实仔细一想,也不难,大家看我把Shop的配置改成下面这样
[java] view plaincopy在CODE上查看代码片派生到我的代码片
Map<String, Object> map=new HashMap<String, Object>(); 
map.put("shops", list); 
map.put("total", total); 
         
JsonConfig jsonConfig = new JsonConfig(); 
jsonConfig.setJsonPropertyFilter(new PropertyFilter() { 
    public boolean apply(Object obj, String name, Object value) { 
    if(value instanceof Shop&&name.equals("shop")){ 
        return true; 
    }else{ 
        return false; 
    } 
   } 
}); 
         
return JSONObject.fromObject(map,jsonConfig); 
这样就可以获取到了,为什么呢?因为这样配置的话,在将Shop里的Set<Staff> staffs转化的时候,我们不过滤;而在将staffs里的每个Staff里的shop转化的时候,我们进行过滤,这样就既解决了死循环问题,又避免了Shop里的staffs被过滤掉的问题。
同理,Staff要这样配置
[java] view plaincopy在CODE上查看代码片派生到我的代码片
Map<String,Object> map=new HashMap<String, Object>(); 
map.put("staffs", list); 
map.put("total", total); 
         
JsonConfig jsonConfig = new JsonConfig();   
jsonConfig.setExcludes(new String[]{"handler","hibernateLazyInitializer"}); 
jsonConfig.setJsonPropertyFilter(new PropertyFilter() { 
       public boolean apply(Object obj, String name, Object value) { 
        if(name.equals("staffs")){ 
            return true; 
        }else{ 
            return false; 
        } 
    } 
}); 
         
return JSONObject.fromObject(map,jsonConfig); 

可是我测试的时候却发现得到的字符串里只有total,staffs没有了?大家可能已经明白了,不仅Shop里的staffs被过滤掉了,map里的staffs也被过滤掉了。解决也很简单,把map.put("staffs",list);改成map.put("list",list);就行了,就是换个名字。

【扩展】到这里,应该能解决大家的问题了。另外还有一种方法也要提一下,
[java] view plaincopy在CODE上查看代码片派生到我的代码片
JsonConfig jsonConfig = new JsonConfig(); 
jsonConfig.setIgnoreDefaultExcludes(false); //设置默认忽略  
jsonConfig.setCycleDetectionStrategy(CycleDetectionStrategy.LENIENT);//设置循环策略为忽略    解决json最头疼的问题 死循环 
jsonConfig.setExcludes(new String[] {"staffs"});//此处是亮点,只要将所需忽略字段加到数组中即可 
这种配置也是比较好理解的,但也要注意属性过滤问题,Shop过滤shop,Staff过滤staffs。

【最后】以上只是我个人的一些理解,如果有什么不正确的地方,还希望大家指正,希望能够帮到大家
分享到:
评论

相关推荐

    JSON死循环解决办法

    在使用Hibernate这样的ORM框架时,由于其懒加载机制,可能会遇到JSON序列化时的死循环问题。这是因为CGLIB动态代理生成的对象在序列化时会递归地尝试序列化所有关联的对象,如果对象之间存在循环引用,就会导致无限...

    PHP 获取JSON json_decode返回NULL解决办法

    PHP 获取JSON json_decode返回NULL解决办法,不小心在返回的json字符串中返回了BOM头的不可见字符,某些编辑器默认会加上BOM头,这样处理才能正确解析json数据

    详解Struts2中json 相互引用死循环解决办法

    总结来说,解决Struts2中JSON相互引用死循环的问题,可以通过配置Struts2的结果类型来控制序列化的属性,或者直接在Action方法上使用注解来标记不需序列化的属性。这两种方式都可以有效地避免无限递归和资源消耗,...

    JSON.rar_JSON Hibernate_extjs_json struts ext_jsp json extjs_str

    在给定的标题"JSON.rar_JSON Hibernate_extjs_json struts ext_jsp json extjs_str"中,我们可以看出这是一个与JSON相关的项目,可能包含了使用JSON进行数据交换的各种技术集成。JSON在这里可能是作为前后端数据传输...

    解决spring mvc JSON 无限死循环

    ### 解决Spring MVC JSON无限死循环问题 在开发基于Spring MVC的应用程序时,经常会遇到将对象转换为JSON格式的需求。然而,在某些情况下,由于对象之间的循环引用,可能会导致JSON序列化过程中出现无限递归的问题...

    java json-lib解决无循环的探索例子

    总之,`java json-lib无限循环`的问题可以通过注解和配置json-lib的`JsonConfig`对象来有效解决。通过忽略特定字段或设置最大递归深度,我们可以确保在处理嵌套数据结构时避免无限循环的风险。在你的项目中,`...

    struts2+extjs2.1+json+hibernate+spring

    上网找资料,都是一些不全的. struts2+extjs2.1+json+hibernate+spring 自己整合的例子. 当中hibernate数据源可以配置自己的.后台输送json 前台接收.

    PHP读取mssql-json数据中文乱码的解决办法.doc

    PHP读取mssql-json数据中文乱码的解决办法.doc

    请求返回json浏览器提示下载问题解决办法

    下载后双击运行,点击确定注册,可以解决后端服务接口返回json数据时浏览器提示下载的问题(下载的文本文件内容为服务端返回的json数据)!

    Struts2+hibernate3+JQuery+ajax+json三级联动

    Struts2、Hibernate3、JQuery、Ajax以及Json是Web开发中的重要技术栈,它们各自在Web应用程序中扮演着不同的角色。在这个“三级联动”项目中,这些技术被巧妙地结合在一起,实现了数据的动态交互和展示。 Struts2是...

    springmvc+spring+hibernate+json文件

    在IT行业中,Spring MVC、Spring、Hibernate和JSON是四个至关重要的技术组件,它们共同构建了高效、可扩展的Java Web应用程序。以下是对这些技术及其整合的详细解释。 **Spring框架** 是一个开源的Java平台,它为...

    springmvc spring hibernate ajax json

    Spring MVC、Spring、Hibernate、Ajax 和 JSON 是五个关键的IT技术,它们在现代Web应用程序开发中扮演着重要的角色。下面将分别对这些技术进行详细解释,并结合它们在Web开发中的应用进行阐述。 **Spring MVC** ...

    struts2+spring+hibernate+jquery+json

    Struts2、Spring、Hibernate、jQuery 和 JSON 是五个在IT行业中极为重要的技术组件,它们各自在Web应用开发中扮演着不同的角色。这篇文章将详细介绍这些技术的整合使用以及它们的功能。 首先,Struts2 是一个基于 ...

    spring mvc 4.0+hibernate4.3+mysql+json用到的jar包

    自己做的项目,基于spring mvc框架,整合了hibernate较新版本,以及需要用到的json包还有用到ueditor前端富文本编辑器的包,十分给力。。组这个包还是花了不少时间。如有问题,欢迎留言交流

    C#接收json数据后中文乱码解决方法

    C#接收Json格式数据,中文出现乱码解决方法

    json2.js json未定义的解决方案!

    JSON2.js是Doug Crockford开发的一个库,主要目的是为了解决老版本JavaScript中对JSON不支持的问题。本文将深入探讨在使用json2.js时遇到"json未定义"错误的解决方案,以及如何正确使用json.parse方法。 首先,...

    extjs+struts2+hibernate+json登录程序

    ExtJS + Struts2 + Hibernate + JSON 登录程序是一个典型的Web开发示例,结合了前端JavaScript框架、MVC框架、持久层框架以及数据传输格式,实现了用户登录功能的前后端交互。下面将详细阐述这些技术及其在登录程序...

    springmvc_hibernate+json lib包

    标题中的"springmvc_hibernate+json lib包"表明这是一个与Spring MVC、Hibernate和JSON相关的库文件集合。在Java开发领域,这三大技术是构建Web应用程序的重要组件。让我们深入了解一下它们各自的功能以及它们如何...

    hibernate-json:Json 用户类型Hibernate

    《深入理解Hibernate-JSON:构建用户类型的JSON存储与查询》 在Java开发中,Hibernate作为一款流行的ORM(对象关系映射)框架,极大地简化了数据库操作。然而,随着JSON数据格式在现代应用中的广泛使用,如何在...

    Json操作及中文乱码解决方案

    解决这个问题通常需要在服务器端设置正确的响应头,如`Content-Type: application/json; charset=utf-8`,声明响应内容是UTF-8编码的JSON。 **三、GsonDemo** 在Java后端,Google的Gson库是常用的JSON处理工具。它...

Global site tag (gtag.js) - Google Analytics