getInteger还是getIntValue,这是一个问题

getInteger()和getInteger()两个方法都是com.alibaba.fastjson.JSONObject中的两个方法,功能类似,从一个JSONObject对象中获取到某个key的value值,区别在于getInteger()返回的是一个integer类型的对象,而getInteger()返回int值,属于基础数据类型。
以此为引……

缘起

在我设计项目框架的时候,在标准项目基类接口中定义了一个非常重要的方法com.funtester.base.interfaces.IBase#isRight,该方法的功能是验证响应结果是否符合规范以及正确性,通常会验证响应体的业务返回值和响应体响应信息。这是一个针对改项目所有接口返回值的验证,不涉及业务。

错误示范

具体的实现方法,举例如下:

/**
 * 验证响应是否合法
 *
 * @param response 接口响应
 * @return
 */
@Override
public boolean isRight(JSONObject response) {
    try {
        return response.getIntValue("code") == 0;
    } catch (Exception e) {
        return false;
    }
}

当然,这是一个错误的示范!

BUG分析

在每个项目中,都会有固定的响应结构,通常来说会在公司层面统一响应结构。改项目的JSON结构响应中最外层的key由code、msg和data构成。该方法的逻辑就是从response最外层获取code的值,然后跟0对比。
但是在某些特殊情况下,响应结构发生变化,最外层已经没有code,如果isRight()接收到的正常的JSONObject对象,那么isRight()返回的依然是true,这与实际情况严重不符。

BUG修复

正确的处理方式如下:

/**
 * 验证响应是否合法
 *
 * @param response 接口响应
 * @return
 */
@Override
public boolean isRight(JSONObject response) {
    try {
        return response.getInteger("code") == 0;
    } catch (Exception e) {
        return false;
    }
}

绝大多数情况下接口响应都是正常的,只有在响应错误(请求失败,HTTPcode错误等等)才会导致响应结构体发生变化,这也是这个BUG一直没出现的原因。这里可以再多加一层验证,对于HTTPcode验证。PS:在FunTester框架中,HTTPcode被保存在response在外层的FunTester的value中。

/**
 * 验证响应是否合法
 *
 * @param response 接口响应
 * @return
 */
@Override
public boolean isRight(JSONObject response) {
    try {
        return response.getInteger("FunTester") == HttpStatus.SC_OK && response.getInteger("code") == 0;
    } catch (Exception e) {
        return false;
    }
}

如果我们还想对这个方法增加一个全局开关来控制验证结果的话,还可以增加一个判断条件。如下:

/**
 * 验证响应是否合法
 *
 * @param response 接口响应
 * @return
 */
@Override
public boolean isRight(JSONObject response) {
    try {
        return Common.VERIFY && response.getInteger("FunTester") == HttpStatus.SC_OK && response.getInteger("code") == 0;
    } catch (Exception e) {
        return false;
    }
}

添加Common.VERIFY条件的好处就是,在性能测试中可以避开一些业务验证。因为在做功能测试和自动化测试的过程中,会经常进行业务上的验证,在验证代码中,可能会涉及一些耗时的操作,这一点在性能测试中是需要避免的。

实践出真知

Demo1

正常使用中测试代码:

public static void main(String[] args) throws IOException {
    JSONObject json = getJson("code=0");
    output(json.getInteger("code") == 0);
    output(json.getIntValue("code") == 0);
}

控制台输出:

INFO-> 当前用户:oker,工作目录:/Users/oker/IdeaProjects/funtester/,系统编码格式:UTF-8,系统Mac OS X版本:10.16
INFO-> true
INFO-> true

Process finished with exit code 0

这里是看不出来两者的区别的。

## Demo2
出现BUG时候的测试代码:
    public static void main(String[] args) throws IOException {
        JSONObject json = getJson("code=0");
        output(json.getIntValue("co") == 0);
        output(null == json.getInteger("co"));
    }

控制台输出:

INFO-> 当前用户:oker,工作目录:/Users/oker/IdeaProjects/funtester/,系统编码格式:UTF-8,系统Mac OS X版本:10.16
INFO-> true
INFO-> true

Process finished with exit code 0

Have Fun ~ Tester !

FunTester,一群有趣的灵魂,腾讯云&Boss认证作者,GDevOps官方合作媒体。


最后: 欢迎大家关注公众号:【 开心螺蛳粉 】,领取一份300页pdf文档的Python自动化测试工程师核心知识点总结!

公众号里大部分资料都是面试时面试官必问的知识点,也包括了很多测试行业常见知识,其中包括了有基础知识、Linux必备、Shell、互联网程序原理、Mysql数据库、抓包工具专题、接口测试工具、测试进阶-Python编程、Web自动化测试、APP自动化测试、接口自动化测试、测试高级持续集成、测试架构开发测试框架、性能测试、安全测试等。

学习上

作为一个软件测试的过来人,我想尽自己最大的努力,帮助每一个伙伴都能顺利找到工作。所以我整理了下面这份资源,现在免费分享给大家,有需要的小伙伴可以关注【公众号:开心螺蛳粉】自提!

软件测试面试文档

我们学习必然是为了找到高薪的工作,下面这些面试题是来自阿里、腾讯、字节等一线互联网大厂最新的面试资料,并且有字节大佬给出了权威的解答,刷完这一套面试资料相信大家都能找到满意的工作。在这里插入图片描述

在这里插入图片描述

行动吧,在路上总比一直观望的要好,未来的你肯定会感谢现在拼搏的自己!如果想学习提升找不到资料,没人答疑解惑时,请及时加入群:1150305204,里面有各种测试开发资料和技术可以一起交流哦。

### 解决方案 `java.lang.NumberFormatException` 是由于尝试将无法解析为数字的字符串转换为数值类型时引发的异常。以下是针对 `JSONObject` 导致该异常的具体分析和解决方案。 #### 1. 异常原因分析 当使用 JSON 库(如 FastJSON 或其他库)解析 JSON 数据并将其映射到 Java 对象时,如果某个字段预期是一个数值类型(如 `int`),但实际上接收到的是一个非数字字符串,则会抛出此异常[^3]。例如: ```java String jsonString = "{\"value\": \"6\u001e\"}"; JSONObject jsonObject = JSONObject.parseObject(jsonString); Integer value = jsonObject.getInteger("value"); // 抛出 NumberFormatException ``` 在此案例中,“6\u001e” 中包含了不可见字符 `\u001e`,这使得整个字符串无法被正确解析为整数。 --- #### 2. 解决方法 ##### 方法一:预处理 JSON 字符串 在解析之前清理掉可能导致错误的非法字符。可以通过正则表达式或其他方式去除多余字符后再进行解析。 ```java import org.json.JSONObject; public class Main { public static void main(String[] args) { String jsonString = "{\"value\": \"6\u001e\"}"; // 清理非法字符 jsonString = jsonString.replaceAll("[^\\x20-\\x7E]", ""); // 去除非可打印 ASCII 字符 JSONObject jsonObject = new JSONObject(jsonString); Integer value = jsonObject.optInt("value", 0); // 使用 optInt 提供默认值以防异常 System.out.println("Parsed Value: " + value); } } ``` 这种方法适用于知道哪些特殊字符可能会干扰解析的情况[^1]。 --- ##### 方法二:捕获异常并提供默认值 通过捕获 `NumberFormatException` 并设置合理的默认值来避免程序崩溃。 ```java import org.json.JSONObject; public class Main { public static void main(String[] args) { String jsonString = "{\"value\": \"6\u001e\"}"; JSONObject jsonObject = new JSONObject(jsonString); try { int value = Integer.parseInt(jsonObject.getString("value")); System.out.println("Parsed Value: " + value); } catch (NumberFormatException e) { System.err.println("Invalid format for 'value': " + e.getMessage()); // 设置默认值或采取其他措施 int defaultValue = 0; System.out.println("Using default value: " + defaultValue); } } } ``` 这种方式适合于不确定输入数据质量的情况下[^2]。 --- ##### 方法三:验证数据结构一致性 确保传入的数据与目标类的定义完全一致。例如,在接收复杂嵌套对象时,应仔细检查每一层的数据类型是否匹配。 假设服务器返回如下响应: ```json { "message": "", "code": 200, "data": "{invalid json}" } ``` 此时直接调用 `toJavaObject()` 可能失败。因此建议先校验再转换: ```java ResponseJson<String> response = new ResponseJson<>(); response.setCode(jsonObject.getIntValue("code")); response.setMessage(jsonObject.optString("message")); // 安全校验 data 是否合法 JSON if (jsonObject.containsKey("data") && !(jsonObject.get("data") instanceof JSONObject)) { throw new IllegalArgumentException("Data field is not a valid JSON object"); } response.setData(jsonObject.getObject("data", String.class)); System.out.println(response.getData()); ``` 这种做法能够有效防止因数据格式不符而导致的运行时错误[^4]。 --- ### 总结 以上三种策略分别从不同角度解决了由 `JSONObject` 引发的 `java.lang.NumberFormatException` 问题。实际应用中可根据具体情况选择最适合的方式加以应对。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值