JSONException: There is a cycle in the hierarchy!解决Json死循环问题

本文探讨如何解决ExtJS GridPanel显示包含外键数据时的JSON转换问题,并介绍使用Hibernate进行数据操作时如何避免懒加载异常。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

{
相信大部分用过Ext的GridPanel显示数据的朋友都遇到过外键数据的问题,首先是转换格式,将数据库里查出来的List集合要转换为JSON格式,如果只是用

JSONObject json = new JSONObject();    

JSONArray dataArray = JSONArray.fromObject(list);

如果list中的数据都是普通数据类型的那能正常转换,但很少这种情况,大部分的数据都存在外键的数据,比如有个News类,他可能会有个发布人User的外键,那么用以上的方法转换是不行,将会出现死循环(我去了解到JSON它是这么一种机制,将数据拆分,一致拆分到全都是基本数据类型位置,比如,一个list存放着多个News数据,它是首先将list拆分成多个News,然后将News的多个属性都拆分出来,里面有个User对象,又得再拆分!问题就出现在这了,User里面又有一个News的集合,News里面又有User,因此导致死循环了);当我遇到这个问题时,我到网上搜了下,发现的解决方案是过滤掉外键对象的数据,以下是我抽出出来写在2个类里面;


json解决hibernate中级联对象延迟加载问题net.sf.json.JSONException: org.... 
在hibernate中将pojo转化为json数据格式时,抛出:net.sf.json.JSONException: org.hibernate.LazyInitializationException,通过网上的资料,解决办法如下(未列出所需库文件):

JsonConfig jsonConfig = new JsonConfig(); 
jsonConfig .setJsonPropertyFilter(new PropertyFilter(){

public boolean apply(Object source, String name, Object value){

if (name.equals("property1") || name.equals("property2")){// 这里填写需要过滤的属性名

     return true;

}

return false;

}

});

JSONObject jsonModel= JSONArray.fromObject(yourObject, jsonConfig );


通过下列途径也可以将不需要的属性过滤掉:

JsonConfig jsonConfig = new JsonConfig();

jsonConfig .setExcludes( new String[]{ "property1" , "property2" } ) ;

JSONObject jsonModel= JSONArray.fromObject(yourObject, jsonConfig );

}

 

 

{一般情况下都要设置Lazy=true, 这样可以少抓点无用的数据, 提高性能. 但是有的时候需要一次性把关联的数据都抓出来, 
这个时候就需要使用积极加载去避免lazy loading了. 延迟加载用得好, 可以提高性能, 用得不好就会出现N+1, 反而会大大
降低性能. 所以需要具体情况具体分析.
}

{HibernateTemplate提供非常多的常用方法来完成基本的操作,比如通常的增加、删除、修改、查询等操作,Spring 2.0更增加对命名SQL查询的支持,也增加对分页的支持。大部分情况下,使用Hibernate的常规用法,就可完成大多数DAO对象的CRUD操作。下面是HibernateTemplate的常用方法简介:
q      void delete(Object entity):删除指定持久化实例
q      deleteAll(Collection entities):删除集合内全部持久化类实例
q      find(String queryString):根据HQL查询字符串来返回实例集合
q      findByNamedQuery(String queryName):根据命名查询返回实例集合
q      get(Class entityClass, Serializable id):根据主键加载特定持久化类的实例
q      save(Object entity):保存新的实例
q      saveOrUpdate(Object entity):根据实例状态,选择保存或者更新
q      update(Object entity):更新实例的状态,要求entity是持久状态
q      setMaxResults(int maxResults):设置分页的大小
}
getHibernateTemplate 与 getSession有什么的区别
{
getHibernateTemplate已经封装好了一些基本的方法,可以直接去用,也就是template嘛, 
而getSession只是获取一个数据工厂的session,然后大部分方法都需要自己写,加hql语句,然后用query方法执行

谈不上什么优点缺点,类似添加删除更新这样的可以直接用getHibernateTemplate而大部分带条件查询的就需要用getSession自己写了
&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&& 
主流技术到底就是主流技术,效率就不一般。
Hibernate封装了对数据库的例行操作,比单纯的jdbc的DAO,开发效率要高很多了。而Springframework对Hibernate的操作又进行了进一步的包装,又将开发效率提升不少。下面的例子是Spring自己给的petclinic的样本程序。
原来b/s架构的大量数据库操作可以这么轻松搞定。

HibernateTemplate 提供非常多的常用方法来完成基本的操作,比如通常的增加、删除、修改、查询等操作,Spring 2.0更增加对命名SQL查询的支持,也增加对分页的支持。大部分情况下,使用Hibernate的常规用法,就可完成大多数DAO对象的CRUD操作。 下面是HibernateTemplate的常用方法简介: 
q void delete(Object entity):删除指定持久化实例 
q deleteAll(Collection entities):删除集合内全部持久化类实例 
q find(String queryString):根据HQL查询字符串来返回实例集合 
q findByNamedQuery(String queryName):根据命名查询返回实例集合 
q get(Class entityClass, Serializable id):根据主键加载特定持久化类的实例 
q save(Object entity):保存新的实例 
q saveOrUpdate(Object entity):根据实例状态,选择保存或者更新 
q update(Object entity):更新实例的状态,要求entity是持久状态 
q setMaxResults(int maxResults):设置分页的大小 
*********************************************
在前一篇日志中自动生成hibernate配置文件的时候,会在dao层用到getSession()方法来操作数据库记录,但是他还有个方法getHibernateTemplate(),这两个方法究竟有什么区别呢?

1.使用getSession()方法你只要继承 sessionFactory,而使用getHibernateTemplate()方法必须继承HibernateDaoSupport当然包括 sessionFactory,这点区别都不是特别重要的,下面这些区别就很重要了

2.getSession()方法是没有经过spring包装 的,spring会把最原始的session给你,在使用完之后必须自己调用相应的close方法,而且也不会对声明式事务进行相应的管理,一旦没有及时 关闭连接,就会导致数据库连接池的连接数溢出,getHibernateTemplate()方法是经过spring封装的,例如添加相应的声明式事务管 理,由spring管理相应的连接。

在实际的使用过程中发现的确getHibernateTemplate()比getSession()方法要好很多,但是有些方法在getHibernateTemplate()并没有提供,这时我们用HibernateCallback 回调的方法管理数据库.

例如如下代码:

/**
     * 使用 hql 语句进行操作
     * @param hql HSQL 查询语句(使用回调函数访问外部变量,必须是final的)
     * @param offset 开始取数据的下标
    * @param length 读取数据记录数
    * @return List 结果集
*/
public List getListForPage ( final String hql , final int offset , final int length ) {

              List list = getHibernateTemplate().executeFind ( new HibernateCallback ( ) {
                            public Object doInHibernate ( Session session ) throws HibernateException, SQLException {
                                            Query query = session.createQuery ( hql ) ;
                                            query.setFirstResult ( offset ) ;
                                            query.setMaxResults ( length ) ;
                                            List list = query.list ( ) ;
                                            return list ;
                           }
               }) ;
               return list ;
}

}

}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值