在SSH框架中,多个持久层对象相互引用,从而引发分页查询中,查询所得的持久化对象转换为JSON字符串报错的原因及解决方案

版权声明:本文为博主原创文章,未经博主允许不得转载。 https://blog.csdn.net/tianjindong0804/article/details/82952682

引言:

使用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的使用方法,博主在另一篇文章中详细介绍了:

 EasyUI框架中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对象:

扫描二维码关注公众号,回复: 3461129 查看本文章
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>

 

猜你喜欢

转载自blog.csdn.net/tianjindong0804/article/details/82952682
今日推荐