[Java] po类解析json'字符串异常 "$ref":"$"

本文介绍了解决阿里Fastjson序列化过程中出现的循环引用问题的方法。通过使用JSON.toJSONStringWithDateFormat方法并设置SerializerFeature.DisableCircularReferenceDetect参数,可以有效避免生成的JSON字符串中出现$ref:$这样的标记。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

在使用阿里的fastjson将实体类解析成json字符串时,发现返回至浏览器时有部分元素变成了"$ref":"$"这样。在网上查了一下,发现是

这样使用就可以了

JSON.toJSONStringWithDateFormat(0,SerializerFeature.DisableCircularReferenceDetect);  

其中:SerializerFeature.DisableCircularReferenceDetect就是禁止循环引用的方案。

参考:

https://blog.youkuaiyun.com/u013185616/article/details/52799166

public class ReadFileContentService { private final ObjectMapper objectMapper = new ObjectMapper(); private Tesseract tesseract; @PostConstruct public void initOcrEngine() { tesseract = new Tesseract(); try { //语言包路径和支持语言 tesseract.setDatapath("D:\\maven_use\\lingxi-lhc\\lingxi-ai-extend\\lingxi-ai-comparison\\src\\main\\resources\\tessdata"); tesseract.setLanguage("eng+chi_sim"); tesseract.setPageSegMode(6); // 自动页面分割 tesseract.setOcrEngineMode(1); // LSTM引擎 } catch (Exception e) { throw new RuntimeException("OCR引擎初始化失败: " + e.getMessage(), e); } } /** * 支持PDF和图片 */ public String extractContent(MultipartFile file) { String contentType = file.getContentType(); String fileName = file.getOriginalFilename().toLowerCase(); if (contentType == null) { return "不支持的文件: " + contentType; } if (fileName.endsWith(".pdf")) { return readPdfText(file); } return extractImageText(file); } /** * 读取PDF文本内容 * * @param file * @return */ public String readPdfText(MultipartFile file) { try (PDDocument doc = PDDocument.load(file.getInputStream())) { PDFTextStripper stripper = new PDFTextStripper(); // 设置行分隔符 stripper.setLineSeparator("\n"); // 设置字符间距 stripper.setSortByPosition(true); String rawText = stripper.getText(doc); System.out.println("内容" + rawText); return rawText.trim(); } catch (Exception e) { return MessageUtils.message("file.red.pdf.error"); } } /** * OCR识别图片内容 */ private String extractImageText(MultipartFile file) { try { // 创建临时文件 Path tempFile = Files.createTempFile("ocr_", getFileExtension(file.getOriginalFilename())); Files.copy(file.getInputStream(), tempFile, StandardCopyOption.REPLACE_EXISTING); // 执行OCR识别 File imageFile = tempFile.toFile(); String result = tesseract.doOCR(imageFile) .replaceAll("\\s+", " ").trim(); System.out.println("读取的内容" + result); // 清理临时文件 Files.deleteIfExists(tempFile); return result; } catch (Exception e) { return "OCR处理失败: " + e.getMessage(); } } private String getFileExtension(String filename) { if (filename == null) return ".tmp"; int dotIndex = filename.lastIndexOf('.'); return (dotIndex == -1) ? ".tmp" : filename.substring(dotIndex); } public JsonNode parseJson(String jsonContent) throws Exception { return this.objectMapper.readTree(jsonContent); } public List<ValidationResult> compareContent(String pdfText, JsonNode jsonConfig) { List<ValidationResult> results = new ArrayList<>(); // 去除读取内容中的多余空格 pdfText = pdfText.replaceAll("\\s+", ""); // 处理JSON结构(支持单个对象或数组) JsonNode dataNode; if (jsonConfig.isArray() && jsonConfig.size() > 0) { dataNode = jsonConfig.get(0); } else if (jsonConfig.isObject()) { dataNode = jsonConfig; } else { results.add(new ValidationResult("ERROR", "JSON格式错误", "期望一个对象或包含对象的数组", "实际格式不匹配", false)); return results; } // 动态定义地址字段列表 Set<String> addressFields = new HashSet<>(); // 字段直接匹配 checkNonAddressFields(pdfText, dataNode, results, addressFields); // 连续匹配 checkAddressFields(pdfText, dataNode, results, addressFields); return results; } /** * 检查 JSON 中非地址字段是否严格存在于 PDF 文本中 */ private void checkNonAddressFields(String pdfText, JsonNode jsonConfig, List<ValidationResult> results, Set<String> addressFields) { Iterator<Map.Entry<String, JsonNode>> fields = jsonConfig.fields(); while (fields.hasNext()) { Map.Entry<String, JsonNode> entry = fields.next(); String fieldName = entry.getKey(); JsonNode valueNode = entry.getValue(); if (valueNode.isValueNode() && !addressFields.contains(fieldName)) { //去除多余空格 String expectedValue = valueNode.asText().trim().replaceAll("\\s+", ""); if (expectedValue.isEmpty()) continue; // 直接进行字符串匹配 boolean found = pdfText.contains(expectedValue); results.add(new ValidationResult( "FIELD", fieldName, expectedValue, found ? "Found" : "Not Found", found )); } } } /** * 检查 JSON 中地址字段是否严格存在于 PDF 文本中 */ private void checkAddressFields(String pdfText, JsonNode jsonConfig, List<ValidationResult> results, Set<String> addressFields) { // HanLP分词 List<Term> terms = HanLP.segment(pdfText); List<String> addressParts = new ArrayList<>(); for (Term term : terms) { String word = term.word; if (word.matches("\\d{5,7}")) { addressParts.add(word); } else if (term.nature.toString().startsWith("ns")) { addressParts.add(word); } } // 遍历 JSON 配置中的地址字段 Iterator<Map.Entry<String, JsonNode>> fields = jsonConfig.fields(); while (fields.hasNext()) { Map.Entry<String, JsonNode> entry = fields.next(); String fieldName = entry.getKey(); JsonNode valueNode = entry.getValue(); if (valueNode.isValueNode() && addressFields.contains(fieldName)) { //去除多余空格 String expectedValue = valueNode.asText().trim().replaceAll("\\s+", ""); if (expectedValue.isEmpty()) continue; boolean found = false; for (String part : addressParts) { if (part.equals(expectedValue)) { found = true; break; } } results.add(new ValidationResult( "FIELD", fieldName, expectedValue, found ? "Found" : "Not Found", found )); } } } public JsonNode parsePipeSeparatedDataToJson(String inputData) throws Exception { Map<String, String> dataMap = parsePipeSeparatedData(inputData); return objectMapper.valueToTree(dataMap); } // 解析以 | 分隔的键值对数据 public Map<String, String> parsePipeSeparatedData(String inputData) { Map<String, String> dataMap = new HashMap<>(); String[] lines = inputData.split("\n"); String[] headers = lines[0].split("\\|"); String[] values = lines[1].split("\\|"); for (int i = 0; i < headers.length; i++) { dataMap.put(headers[i], values[i]); } return dataMap; } // 判断是否是以 | 分隔的数据 public boolean isPipeSeparatedData(String inputData) { return inputData.contains("|"); } }这个是我的代码 public R compare(@RequestPart("File") MultipartFile file, @RequestPart("jsonContent") String jsonContent) { // 读取 PDF 文本 String pdfText = compareService.extractContent(file); // 解析 JSON 配置 JsonNode jsonConfig = null; try { if (compareService.isPipeSeparatedData(jsonContent)) { jsonConfig = compareService.parsePipeSeparatedDataToJson(jsonContent); System.out.println("数据"+jsonContent); } else { jsonConfig = compareService.parseJson(jsonContent); } } catch (Exception e) { return R.fail(MessageUtils.message("failed.convert.json")); } // 执行对比校验 List<ValidationResult> results = compareService.compareContent(pdfText, jsonConfig); // 返回没有匹配成功的数据 List<ValidationResult> failedResults = new ArrayList<>(); for (ValidationResult result : results) { if (!result.isValid()) { failedResults.add(result); } } return failedResults.isEmpty() ? R.ok("条件符合规范") : R.ok(failedResults); }这个是我的接口 这个是我读取pdf里面的文件的内容 内容Delivery Note Apple Computer Trading *QBA7074052* (Shanghai)Co Ltd 1249 Century Avenue Tower 3, Page No : 1 of 1 Pudong new district, Delivery Reference: QBA7074052 shanghai 200122, China Delivery Date: 13-MAY-2025 Web Ref#: 2365330821 Carrier: XDESC Mode: ROAD Sold To: Ship To: 深圳市天联终端有限公司 深圳市天联终端有限公司上海库 前湾一路1号A栋201室 嘉定区兴邦路1166号 深圳广东518000 上海上海201800 中国 中国 Cust PO No. Cust PO Line # Sales Order# Delivery No# Item No. Part No. Description Qty Shipped 4500304123-TLZD AAA3871466 QBA7074052 000400 MTLH3CH/A IPHONE 15 GREEN 128GB-CHN 20 -03 4500304123-TLZD AAA3871466 QBA7074052 000620 MTLG3CH/A IPHONE 15 BLUE 128GB-CHN 160 -03 Total Quantity: 180 Total Packages:18 Total Weight:73.94 KGS Cube:0.22 CBM Damaged Boxes · Part Number: · Quantity: IMPORTANT REMINDER Please verify that all part numbes(s) and quantity are as started above. We confirm that: 1) We received ___________ cartons/boxes as started above. 2) We received the goods as started above. The goods received are inspected and found to be in good order and condition. Received By __________________ Received Date ____________________ (Signature & Company stamp) 这个是我的传的内容 数据"filE_CONTENT": "PALLET_ID|KNBOXNO|CARRIER|COC|CTRY|HAWB|PO|ORIGIN|INVOICENO|CARTONS|SHIPID|SHIP_DATE|TEL|SSCC|RETURN_TO1|RETURN_TO2|RETURN_TO3|RETURN_TO4|RETURN_TO5|RETURN_TO6|RETURN_TO7|RETURN_TO8|SHIP_TO1|SHIP_TO2|SHIP_TO3|SHIP_TO4|SHIP_TO5|SHIP_TO6|SHIP_TO7|SHIP_TO8|LINEITEM1|MPN1|QTY1|LINEITEM2|MPN2|QTY2|LINEITEM3|MPN3|QTY3|\nFO2501000233P0002||DGF-AD|NL|NL|8VG8286|0638138589|PVG|8VG8286|61/84|73292885370002|06/01/2025|00000000|001959499064098506|ADI EMEIA Logistics|Inbound Department|||||||PEGATRON CORPORATION|c/o DP World Logistics Netherlands BV|Van Hilststraat 23|5145 RK Waalwijk,Netherlands|5145 RK Waalwijk Netherlands||||00010|MLPF3AA/A|10|"\n 后的是要打印的内容| 是来区分不同的参数 我如果是json的话可以比较和校验 我传这种的话就没有校验 而是直接返回{ "code": 200, "msg": "条件符合规范", "data": null } 没有把错误的都比较出来
最新发布
07-12
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值