首页 诗词 字典 板报 句子 名言 友答 励志 学校 网站地图
当前位置: 首页 > 教程频道 > 网站开发 > JavaScript >

Json-lib jQuery ext与 hibernate 共同使用死循环有关问题解决方案

2012-11-03 
Json-libjQuery ext与 hibernate 共同使用死循环问题解决方案报错net.sf.json.JSONException: There is a

Json-lib jQuery ext与 hibernate 共同使用死循环问题解决方案

报错

net.sf.json.JSONException: There is a cycle in the hierarchy!net.sf.json.util.CycleDetectionStrategy$StrictCycleDetectionStrategy.handleRepeatedReferenceAsObject(CycleDetectionStrategy.java:97)net.sf.json.JSONObject._fromBean(JSONObject.java:833)net.sf.json.JSONObject.fromObject(JSONObject.java:168)net.sf.json.AbstractJSON._processValue(AbstractJSON.java:265)net.sf.json.JSONArray._processValue(JSONArray.java:2514)net.sf.json.JSONArray.processValue(JSONArray.java:2539)net.sf.json.JSONArray.addValue(JSONArray.java:2526)net.sf.json.JSONArray._fromCollection(JSONArray.java:1057)net.sf.json.JSONArray.fromObject(JSONArray.java:123)net.sf.json.AbstractJSON._processValue(AbstractJSON.java:237)net.sf.json.JSONObject._processValue(JSONObject.java:2808)net.sf.json.JSONObject.processValue(JSONObject.java:2874)net.sf.json.JSONObject.setInternal(JSONObject.java:2889)net.sf.json.JSONObject.setValue(JSONObject.java:1577)net.sf.json.JSONObject._fromBean(JSONObject.java:934)net.sf.json.JSONObject.fromObject(JSONObject.java:168)net.sf.json.AbstractJSON._processValue(AbstractJSON.java:265)net.sf.json.JSONObject._processValue(JSONObject.java:2808)net.sf.json.JSONObject.processValue(JSONObject.java:2874)net.sf.json.JSONObject.setInternal(JSONObject.java:2889)net.sf.json.JSONObject.setValue(JSONObject.java:1577)net.sf.json.JSONObject._fromBean(JSONObject.java:934)net.sf.json.JSONObject.fromObject(JSONObject.java:168)net.sf.json.AbstractJSON._processValue(AbstractJSON.java:265)

?

?

?

hibernate使用CGLIB把POJO的domain对象动态代理,实现它的魔法,但是给JSON的序列化带来了麻烦,因为JSON无法对lazy的属性进行序列化。有以下的四个方法可以解决hibernate的序列化问题

    domain类实现JSONString接口
    建立JsonConfig实例,并配置属性排除列表用属性过滤器写一个自定义的JsonBeanProcessor

1. 实现JSONString接口是侵入性最强的方法

public class Person implements JSONString {   private String name;   private String lastname;   private Address address;    // getters & setters    public String toJSONString() {      return "{name:'"+name+"',lastname:'"+lastname+"'}";   }}
<span id="more-724"></span>

2.第二种方法通过jsonconfig实例,对包含和需要排除的属性进行方便添加删除

public class Person {   private String name;   private String lastname;   private Address address;    // getters &amp; setters} JsonConfig jsonConfig = new JsonConfig();jsonConfig.setExclusions( new String[]{ "address" } );Person bean = /* initialize */;JSON json = JSONSerializer.toJSON( bean, jsonConfig );

注意:这种方法不区分目标类,就是说如果有2个bean当中都存在“address”属性,那么采用这种方法,这两个bean中的address属性都将被排除

3. 使用propertyFilter可以允许同时对需要排除的属性和类进行控制,这种控制还可以是双向的,也可以应用到json字符串到java对象

public class Person {   private String name;   private String lastname;   private Address address;    // getters &amp; setters} JsonConfig jsonConfig = new JsonConfig();jsonConfig.setJsonPropertyFilter( new PropertyFilter(){   public boolean apply( Object source, String name, Object value ){      // return true to skip name      return source instanceof Person &amp;&amp; name.equals("address");   }});Person bean = /* initialize */;JSON json = JSONSerializer.toJSON( bean, jsonConfig )

4. 最后来看JsonBeanProcessor,这种方式和实现JsonString很类似,返回一个代表原来的domain类的合法JSONOBJECT

public class Person {   private String name;   private String lastname;   private Address address;    // getters &amp; setters} JsonConfig jsonConfig = new JsonConfig();jsonConfig.registerJsonBeanProcessor( Person.class,   new JsonBeanProcessor(){      public JSONObject processBean( Object bean, JsonConfig jsonConfig ){         if( !(bean instanceof Person) ){            return new JSONObject(true);         }         Person person = (Person) bean;         return new JSONObject()            .element( "name", person.getName() )            .element( "lastname", person.getLastname() );      }});Person bean = /* initialize */;JSON json = JSONSerializer.toJSON( bean, jsonConfig );

仔细查了一下发现是hibernate主外键关联的错,后来就想下json源代码下来看,发现大费周章都没搞到json源码,还是老办法反编译瞅瞅,发现JSONArray根据判断取得的不同类型调用相应的方法,

if (object instanceof Collection)
??? return _fromCollection((Collection)object, jsonConfig);

而我从hibernate那得到的是list,所以去调用了_fromCollection方法,而里面的方法发现一个问题:该方法会不断的拆开实体属性,直到没有为止,而我的ContactGroup里有两个属性用于自身关联

private Set contactGroups = new HashSet(0);
private Set contactGroupPersons = new HashSet(0);


也就是说主外键自身关联的是个死循环,那怎么才能不让他出现这种情况呢,应该有个配置的参数后者终止循环的地方吧,查看发现,jsonConfig,呵呵,config应该是配置参数吧,参看JsonConfig看见巨多的属性,有点晕PropertyFilter ,不提了,看了老半天,发现了一个属性PropertyFilter,PropertyFilter 是一个interface,代码如下:

public interface PropertyFilter
{

public abstract boolean apply(Object obj, String s, Object obj1);
}

也就是说我可以通过这个方法过滤掉List里的相应属性,只要让它返回true就可过滤掉,……,有点悬……我们重写一下这个方法:

JsonConfig cfg = new JsonConfig();
??? cfg.setJsonPropertyFilter(new PropertyFilter()
??? {
??? ???? public boolean apply(Object source, String name, Object value) {
???? ????? if(name.equals("contactGroups")||name.equals("contactGroupPersons")) {
????? ?????? return true;
???? ????? } else {
????? ?????? return false;
??? ????? }
??? ??? }
??? ?? });

将hibernate产生的实体bean中的contactGroups和contactGroupPersons过滤掉就OK了!

然后调用JSONArray.fromObject(mychildren,cfg); mychildren是hibernate返回的list。

Okar,就这样,继续做项目……

热点排行
Bad Request.