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的序列化问题

  1. domain类实现JSONString接口
  2. 建立JsonConfig实例,并配置属性排除列表
  3. 用属性过滤器
  4. 写一个自定义的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,就这样,继续做项目……

转载于:https://my.oschina.net/usenrong/blog/197860

猜你喜欢

转载自blog.csdn.net/weixin_34292959/article/details/92028898