返回数据存在"$ref":"$.data.*** "的问题。

记录一次 JSONArray 引发的惨案!!!

参考文章:https://blog.youkuaiyun.com/ljc20090913/article/details/81662690
参考文章:https://www.jianshu.com/p/f22f98503ead

今日领导给我分了一个bug。发现我接口返回的数据中存在 “ref":"ref":"ref":".data.*** " 和JSONArray出现这种情况 {“ref”:”.nameArr[0]"}的问题。百思不得其解,后端打断点数据正常,但是前端就是不行。
后来网上搜到一篇文章说是:使用FastJson的JSONArray类型作为返回数据,当向JSONArray对象中添加JSONObject对象,而JSONObject对象中包含相同的节点数据时,FastJson会防止返回数据栈溢出的问题,自动将JSONArray中相同的节点数据使用引用方式代替

有问题的代码

 ResponseUtils.renderJson(response, JSONObject.toJSONStringWithDateFormat(apiResult, ApiResult.DATE_FORMATE));
public static void render(HttpServletResponse response, String contentType,
			String text) {
		response.setContentType(contentType);
		response.setHeader("Pragma", "No-cache");
		response.setHeader("Cache-Control", "no-cache");
		response.setDateHeader("Expires", 0);
		try {
			response.getWriter().write(text);
		} catch (IOException e) {
			log.error(e.getMessage(), e);
		}
	}

项目中的数据是返回一个list集合,但是集合中的实体类中有一个字段是对其它对象的引用。
解决方法,将返回数据改为java对象

解决方式一:将返回的字符串改为java实体类

 String resultStr = JSONObject.toJSONStringWithDateFormat(apiResult, ApiResult.DATE_FORMATE);
        JSONObject jsonObject = JSONObject.parseObject(resultStr);
        apiResult.setData(jsonObject.getJSONObject("data"));
        apiResult.setSuccess(jsonObject.getBoolean("success"));
        return apiResult;

解决方式二:
真正原因: 发现是因为使用FastJson的JSONArray类型作为返回数据,当向JSONArray对象中添加JSONObject对象,而JSONObject对象中包含相同的节点数据时,FastJson会防止返回数据栈溢出的问题,自动将JSONArray中相同的节点数据使用引用方式代替

解决方案(禁用循环引用)
强大的 FastJson 为我们提供了相关的配置参数来禁用循环引用,示例代码如下:

JSONArray jsonArrayUserNum = JSONArray.parseArray(JSON.toJSONString(resultJsonArr, SerializerFeature.DisableCircularReferenceDetect));

流程:禁止循环引用–>转化为 json 字符串–>再降 json 串转为 JSONArray–>存入新的 JSONArray。
之前就遇到过这个问题,一开始以为是深拷贝浅拷贝的问题,今天终于是找到原因了,解决问题了

### 解析 JSON 中 `$ref` 引用路径 在处理 JSON 数据时,遇到 `$ref` 是一种常见的引用机制。它允许通过相对或绝对路径来引用其他部分的数据结构,从而减少冗余数据并提高效率。 #### 路径解析逻辑 对于路径 `$.itemList[0].cntrList[0]` 的解析过程可以分为以下几个方面: 1. **根节点 (`$`)** - 符号 `$` 表示当前 JSON 对象的根节点[^1]。 2. **成员访问 (`.itemList`)** - 使用点操作符 `.itemList` 来访问根对象下的 `itemList` 属性。 3. **数组索引 (`[0]`)** - 数组可以通过方括号 `[0]` 访问其第一个元素。 4. **嵌套属性访问 (`.cntrList[0]`)** - 继续沿着树状结构向下访问子属性 `cntrList` 和它的第一个元素。 #### 实现代码示例 以下是实现该路径解析的一个简单 JavaScript 示例: ```javascript function resolveRef(jsonObj, refPath) { const pathParts = refPath.replace(/^\$\./, '').split('.'); let current = jsonObj; try { for (const part of pathParts) { if (part.includes('[')) { // 处理数组索引情况 const match = part.match(/([a-zA-Z_]+)\[(\d+)\]/); if (!match || !current[match[1]] || !(Array.isArray(current[match[1]]) && current[match[1]][parseInt(match[2])])) { throw new Error(`Invalid reference at ${part}`); } current = current[match[1]][parseInt(match[2])]; } else { // 普通键名访问 if (!(part in current)) { throw new Error(`Key "${part}" not found`); } current = current[part]; } } return current; } catch (e) { console.error(e.message); return null; } } // 测试数据 const data = { itemList: [ { cntrList: ["A", "B"] }, {} ] }; console.log(resolveRef(data, "$.itemList[0].cntrList[0]")); // 输出 "A" ``` 此代码片段定义了一个通用函数 `resolveRef`,用于解析指定的 `$ref` 路径,并返回对应的值。 #### 关于循环引用的注意事项 当禁用循环引用检测时,可能会导致栈溢出错误(`StackOverflowError`)。因此,在实际应用中应谨慎使用全局关闭引用检测的功能[^2]。如果确实需要局部控制,则建议仅针对特定字段或方法进行调整。 另外需要注意的是,某些框架如 Vue.js 提供了自己的方式管理 DOM 元素引用(即 `ref`),这与 JSON Schema 中使用的 `$ref` 不同[^3]。两者不应混淆。
评论 1
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值