引言:
使用EasyUI框架的小伙伴们都知道,在使用datagrid(数据表格)时,会传入如一个URL,用于请求数据表格中的所显示的数据:
<!--创建一个空表格-->
<table id="mytable"></table>
<script type="text/javascript">
$(function() {
//页面加载完成后,创建数据表格datagrid
$("#mytable").datagrid(
{
//定义标题行所有的列,注意这是一个二维数组
columns : [ [ {
title : '编号',
field : 'id',
checkbox : true
}....... ] ],
//指定数据表格发送ajax请求的地址
url : '${pageContext.request.contextPath }/subareaAction_pageQuery'
});
});
</script>
而后台相应的是一个JSON字符串,具体datagrid的使用方法,博主在另一篇文章中详细介绍了:
既然后端需要响应的是一个JSON字符串自然需要用到JSON的转换工具,JSON转换工具有多种,我们这里使用Jsonlib。通过这个工具我们将在web层中将Service层返回的持久层对象转换成JSON格式的字符串,然后响应给客户端。那么在转换成JSON字符串过程中会出现一个问题,需要我们注意!
问题描述:
我们这里给出两个持久层对象:
Region对象代表区域对象,一个区域中有多个分区;但一个分区只对应一个区域。
Subarea对象(省略Get和Set方法):
public class Subarea implements java.io.Serializable {
// Fields
private String id;
private Decidedzone decidedzone;
private Region region;
private String addresskey;
private String startnum;
private String endnum;
private String single;
private String position;
}
Region对象:
public class Region implements java.io.Serializable {
// Fields
private String id;
private String province;
private String city;
private String district;
private String postcode;
private String shortcode;
private String citycode;
private Set subareas = new HashSet();
}
如果我们客户端需要需要获得所有Subarea的数据,此时我们会将所有Subarea对象查询出来并返回给web层,在web层中我们会将LIst<Subarea>对象通过Jsonlib工具转换成JSON字符串。但如果试图直接用 JSONArray.fromObject(list) 方法将查询出来的List对象转换成JSON字符串会报错。
问题分析与解决:
之所以出现上述问题,是因为在每个Subarea对象中有一个Region对象,而Region对象中又会包含Subarea本身,所以在这里会出现Java对象转JSON的异常。
那我们如何将解决这个问题呢?这里我么需要分两种情况
情况一:前端页面不需要Subarea中Region对象的数据
这个情况很好解决,我们只需要在生成JSON对象时排除Region对象所对应的属性即可。
public class SubareaAction extends BaseAction<Subarea>{
public String pageQuery() {
subareeaService.pageQuery(pageBean);
JsonConfig config = new JsonConfig();
config.setExcludes(new String[] {"currentPage","pageSize","detachedCriteria","region"});
JSONObject json = JSONObject.fromObject(obj, config);
ServletActionContext.getResponse().setContentType("text/json;charset=utf-8");
try {
ServletActionContext.getResponse().getWriter()
.write(json.toString());
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
return null;
}
}
情况二:前端页面需要Region中的数据
有些小伙伴可能就会说了,直接排除Region对象的subareas属性不就行了。但这样做还不够,因为Hibernate默认配置中,关联对象采用延迟加载策略。Subarea对象中的Region实际上是一个代理对象。如果Jsonlib尝试将这个代理对象转换成字符串时,同样会报错!
解决步骤:
第一步:在将List<Subarea>转换成JSON字符串时,需要将List中每个Subarea中Region对象的subareas属性排除。
public class SubareaAction extends BaseAction<Subarea>{
public String pageQuery() {
subareeaService.pageQuery(pageBean);
JsonConfig config = new JsonConfig();
config.setExcludes(new String[] {"currentPage","pageSize","detachedCriteria","subareas"});
JSONObject json = JSONObject.fromObject(obj, config);
ServletActionContext.getResponse().setContentType("text/json;charset=utf-8");
try {
ServletActionContext.getResponse().getWriter()
.write(json.toString());
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
return null;
}
}
第二步:在配置文件中关闭关联对象的延迟加载
<hibernate-mapping>
<class name="com.tjd.bos.domain.Subarea" table="bc_subarea" >
<id name="id" type="java.lang.String">
<column name="id" length="32" />
<generator class="uuid" />
</id>
<many-to-one name="decidedzone" class="com.tjd.bos.domain.Decidedzone" fetch="select">
<column name="decidedzone_id" length="32" />
</many-to-one>
//重点就是在这里添加 lazy="false"
<many-to-one lazy="false" name="region" class="com.tjd.bos.domain.Region" fetch="select">
<column name="region_id" length="32" />
</many-to-one>
<property name="addresskey" type="java.lang.String">
<column name="addresskey" length="100" />
</property>
<property name="startnum" type="java.lang.String">
<column name="startnum" length="30" />
</property>
<property name="endnum" type="java.lang.String">
<column name="endnum" length="30" />
</property>
<property name="single" type="java.lang.String">
<column name="single" length="1" />
</property>
<property name="position" type="java.lang.String">
<column name="position" />
</property>
</class>
</hibernate-mapping>