Hibernate中如果出现了级联查询,就可能出现懒加载问题,比如我现在有个Account(管理员)类、Category(商品类别)和Product(商品)类,从左到右都是一对多的关系,而且从右到左都是设置了@ManyToOne(fetch=FetchType.LAZY)。我现在要把商品信息查出来打包成json格式传到前台,我在后台使用查询语句为:
from Product p left join fetch p.category where p.name like:name
这样就可以把Product查出来了,然后Product中的Category也放进去了,但是Category中的Account不是实际对象,是暂时的代理对象,这点很好理解,因为我就查了Product而且只级联了Category,至于Category和Account就根据实际配置了(LAZY)。
现在将查询出来的product放到Map中,然后转成json格式返回到前台肯定会出现懒加载问题,因为在转json的过程中会拿Account对象,但是此时session已经关闭了,所以会报错,有个很直接但是不太好的解决办法就是将Category中的LAZY改成EAGER,这样就能把Account的信息也查出来,但是这样不好。所以我们用另一种办法:在struts.xml中设置一下黑名单,在转json格式的时候使用正则表达式将category中的account过滤掉,就不会去查account对象了,就不会有懒加载问题了。如下:
到这里,应该就没问题了。但是在我的项目中还是报懒加载异常,也就是说我这样配置后根本没有起作用。但是理论上,这样配置后就OK了,就可以正常的把数据打包成json格式传给前台了。这问题困扰了我2天,后来索性先将LAZY改成EAGER,先把项目往下做。
今天我在另一个Hibernate异常中,联系到了这里的异常,解决了!Hibernate中今天我要调用get方法获取商品的信息,无法获取到,后台的控制台没有任何消息,由于我开启了dev模式,前台显示了错误信息:
java.lang.ClassCastException:cn.it.shop.model.Product_$$_javassist_0 cannot be cast to javassist.util.proxy.Proxy</span>
无法转成代理??为啥要转成代理呢?一般不都是代理无法转成实际对象么?于是我上网搜索了一下,这个问题可能是由于项目中的一个javassist的jar包冲突了,我去工程中检查一下,果不其然:
还真的冲突了哟喂……于是删掉struts包中的那个javassist-3.11.0.GA.jar即可,Hibernate这边没错了,可以正常拿出商品信息了。然后我联想到了2天前struts2转json的问题,于是回去将EAGER改回LAZY,问题也没了,也能正常转json了,郁闷,还真是jar包冲突惹的祸。因为当时根本没有报错,只是前台那边我查不到返回的json数据,只知道没有返回json数据,肯定是后台转json出了问题,根据已有的经验,90%是懒加载的问题,但是没想到是jar包冲突惹的祸。
后话:如果jar包没冲突,但是无法转json,那基本上是懒加载惹的祸,利用struts.xml中配置黑名单的方式将懒加载的对象过滤掉的方法很实用,不用修改POJO中的配置,我想把哪些字段转到json中就转哪些,不想就不转,很方便。
_____________________________________________________________________________________________________________________________________________________
-----乐于分享,共同进步!