引言:

本文描述了在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. 知识图谱在RAG中的应用探讨

    在这篇文章中,我们来详细探讨知识图谱(KG)在RAG流程中的具体应用场景. 缘起 关于知识图谱在现在的RAG中能发挥出什么样的作用,之前看了360 刘焕勇的一个分享,简单的提了使用知识图谱增强大模型的 ...

  2. HarmonyOS 电话服务开发指导

    电话服务开发概述 HarmonyOS 电话服务系统提供了一系列的 API 用于拨打电话.获取无线蜂窝网络和 SIM 卡相关信息. 应用可以通过调用 API 来获取当前注册网络名称.网络服务状态.信号强 ...

  3. 重新点亮linux 基本软件————防火墙[一]

    前言 简单介绍一下linux的防火墙. 正文 防火墙分类: 软件防火墙和硬件防火墙 包过:过滤防火墙和应用层防火墙 iptables 的表和链 规则表: filter nat mangle raw f ...

  4. jenkins 持续集成和交付 —— 邮箱服务器配置(九)

    前言 简介邮箱服务器的配置,让jenkins在构建完成后,能有一个邮箱结果通知到我们,这样就不用每次盯着jenkins 看是否完成了. 正文 1.安装插件 安装下面这个插件. Email Extens ...

  5. c# unsafe

    前言 c# unsafe可以让我们做一些测试的危险操作. 正文 打开unsafe 功能 在 Visual Studio 开发环境中设置此编译器选项 打开项目的"属性"页. 单击&q ...

  6. Jenkins集成GitLab的正确姿势,实现Git代码提交触发CI/CD

    ❝ jenkins和gitlab是目前DevOps工具链中最常见的,抛开gitlab-ci不谈,gitlab代码提交触发jenkins流水线是最经典的搭配. 这里就介绍下如何配置实现jenkins和g ...

  7. 力扣1773(java&python)-统计匹配检索规则的物品数量(简单)

    题目: 给你一个数组 items ,其中 items[i] = [typei, colori, namei] ,描述第 i 件物品的类型.颜色以及名称. 另给你一条由两个字符串 ruleKey 和 r ...

  8. 【SIGIR 2022】面向长代码序列的Transformer模型优化方法,提升长代码场景性能

    简介: 论文主导通过引入稀疏自注意力的方式来提高Transformer模型处理长序列的效率和性能 阿里云机器学习平台PAI与华东师范大学高明教授团队合作在SIGIR2022上发表了结构感知的稀疏注意力 ...

  9. 持续定义Saas模式云数据仓库+实时分析

    简介: 从实时分析的价值.场景和数据流程,以及用户对平台能力要求展开,讲述云数据仓库MaxCompute的产品能力优势 ,面对实时分析场景的能力演进要求.进而以实时分析典型场景的全数据流程处理.建模和 ...

  10. Serverless在游戏运营行业进行数据采集分析的最佳实践

    简介: 这个架构不光适用于游戏运营行业,其实任何大数据采集传输的场景都是适用的,目前也已经有很多客户正在基于Serverless的架构跑在生产环境,或者正走在改造Serverless 架构的路上. 众 ...