引言:

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

使用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对象:

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>

Hibernate双向关联导致Java对象转换为JSON字符串时死循环问题的分析与解决方案的更多相关文章

  1. JackSon将java对象转换为JSON字符串

    JackSon可以将java对象转换为JSON字符串,步骤如下: 1.导入JackSon 的jar包 2.创建ObjectMapper对象 3.使用ObjectMapper对象的writeValueA ...

  2. (后端)JackSon将java对象转换为JSON字符串(转)

    转载小金金金丶园友: JackSon可以将java对象转换为JSON字符串,步骤如下: 1.导入JackSon 的jar包 2.创建ObjectMapper对象 3.使用ObjectMapper对象的 ...

  3. Jackson将对象转换为json字符串时,设置默认的时间格式

    maven需要的依赖: <dependency> <groupId>com.fasterxml.jackson.core</groupId> <artifac ...

  4. java对象与json字符串的互相转换

    java对象与json字符串的互相转换 1.采用 net.sf.json.JSONObject maven依赖包: <dependency> <groupId>net.sf.j ...

  5. Java基础97 json插件的使用(java对象和json字符串对象之间的转换)

    1.需要用到的包 2.实例 实体类 people package com.shore.entity; /** * @author DSHORE/2019-4-19 * */ public class ...

  6. struts2:JSON在struts中的应用(JSP页面中将对象转换为JSON字符串提交、JSP页面中获取后台Response返回的JSON对象)

    JSON主要创建如下两种数据对象: 由JSON格式字符串创建,转换成JavaScript的Object对象: 由JSON格式字符串创建,转换成JavaScript的List或数组链表对象. 更多关于J ...

  7. DataTable 对象 转换为Json 字符串

    /// <summary> /// DataTable 对象 转换为Json 字符串 /// </summary> /// <param name="dt&qu ...

  8. java对象与Json字符串之间的转化(fastjson)

    1. 首先引入jar包 在pom.xml文件里加入下面依赖: <dependency> <groupId>com.alibaba</groupId> <art ...

  9. Django 将数据库查出的 QuerySet 对象转换为 json 字符串

    通过Django查询出MySQL数据库的数据,并将查询出的QuerySet 对象转化成 json 字符串. 写这个例子的作用主要是用来为手机端提供接口用,记录一下,以后 说不准 肯定能用到! ---- ...

  10. java对象转为json字符串

    1.使用fastjson开源json工具类库 2.java类未添加get()和set()方法,java对象初始化时,使用fastjson解析,得到的json字符串有时为空{} 3.Java对象转为js ...

随机推荐

  1. 《深入理解Java虚拟机》读书笔记: 虚拟机类加载的时机和过程

    虚拟机类加载的时机和过程 一.类加载的时机 类从被加载到虚拟机内存中开始,到卸载出内存为止,它的整个生命周期包括:加载(Loading).验证(Verification).准备(Preparation ...

  2. Nginx 简介、安装与配置文件详解

    〇.前言 在日常工作中,Nginx 的重要性当然不言而喻. 经常用,但并不意味着精通,还会有很多不清楚的方式和技巧,那么本文就简单汇总下,帮助自己理解. 一.Nginx 简介 1.1 关于 Nginx ...

  3. 初识大数据技术之Hadoop

    先上一张图: 看到这张图,我脑子里出现的第一个东西就是:这货太像旅行商问题了 有限的输入与有限的输出,当输入大于一定数值时,输出趋向于无法计算.... 其实要我说啊,旅行商问题其实没必要管他,因为这个 ...

  4. 使用 Docker 部署 Draw.io 在线流程图系统

    1)介绍 Draw.io GitHub:https://github.com/jgraph/drawio Draw.io 是一款开源的绘制流程图的工具,拥有大量免费素材和模板.程序本身支持中文在内的多 ...

  5. 力扣1098(MySQL)-小众书籍(中等)

    题目: 书籍表 Books: book_id 是这个表的主键 订单表 Orders: order_id 是这个表的主键.book_id 是 Books 表的外键. 问题你需要写一段 SQL 命令,筛选 ...

  6. 力扣306(java)-累加数(中等)

    题目: 累加数 是一个字符串,组成它的数字可以形成累加序列. 一个有效的 累加序列 必须 至少 包含 3 个数.除了最开始的两个数以外,序列中的每个后续数字必须是它之前两个数字之和. 给你一个只包含数 ...

  7. EasyNLP开源|中文NLP+大模型落地,EasyNLP is all you need

    ​简介:EasyNLP背后的技术框架如何设计?未来有哪些规划?今天一起来深入了解. 作者 | 临在.岑鸣.熊兮 来源 | 阿里开发者公众号 一 导读 随着BERT.Megatron.GPT-3等预训练 ...

  8. 重磅发布 阿里云数据中台全新产品DataTrust聚焦企业数据安全保障

    简介: DataTrust(隐私增强计算产品)是基于阿里云底层多项基础安全能力,经过阿里云数据中台丰富的客户业务实践,构建的一款为企业数据安全流通的产品. 随着包括零售.制造.金融等多行业数字化转型加 ...

  9. 冷热分离之OTS表格存储实战

    简介: 为什么要冷热分离由于2020疫情的原因,在线教育行业提前被大家所重视,钉钉教育已经服务超过21万所学校.700万教师和1.4亿学生用户,每天大量的教育数据产生.整体数据量:随着时间的积累,数据 ...

  10. KubeVela + KEDA:为应用带来“与生俱来”的弹性伸缩能力

    简介: 在这篇博文中,我们将简要解释需要考虑的领域,KEDA 如何使应用自动伸缩变得简单,以及为什么阿里云企业分布式应用服务(EDAS)在 KEDA 上完全标准化. 联合作者 | Yan Xun,阿里 ...