Java数字输入验证需分四层:字符串结构校验(如去空、首字符检查)、类型转换(避免异常)、数值范围控制(如0~150)、业务语义关联(如时间先后)。推荐用Optional封装工具方法统一处理。

Java中验证数字输入的合法性,核心是区分“字符串能否转为数字”和“数字是否符合业务规则”两个层次。不能只靠try-catch简单包裹Integer.parseInt()就认为万事大吉——比如“123.45”、“+123”、“ 42 ”、“NaN”、“1e5”这些输入,不同场景下合法与否完全不同。
基础类型转换前的预处理与边界判断
直接调用parseXxx()容易抛出NumberFormatException,但异常成本高、信息粗粒度。更稳妥的做法是先做轻量级校验:
- 用
String.trim().isEmpty()排除空或纯空白输入 - 检查首字符:是否为
'-'、'+'或数字(避免“abc123”被部分识别) - 遍历非首字符,确保全是ASCII数字(
c >= '0' && c ),拒绝小数点、e、下划线等 - 对整数,提前比对长度——如
int最多10位十进制数(-2147483648 ~ 2147483647),超长可直接返回非法
按需选择解析方式:Integer、BigDecimal还是正则?
不同数字类型对应不同验证逻辑:
-
整数(无小数):优先用
Integer.valueOf(str)(自动缓存常用值),配合try-catch捕获溢出;若范围更大,改用Long.parseLong()或BigInteger -
带精度的十进制数:必须用
BigDecimal,避免Double.parseDouble()引入浮点误差(如"0.1 + 0.2 ≠ 0.3");同时注意new BigDecimal("0.10")保留两位小数,而Double.toString(0.1)可能生成"0.10000000000000001" -
格式强约束(如手机号、邮编、金额):用正则更直观,例如
"^\\d{11}$"校验11位手机号,"^\\d+(\\.\\d{2})?$"匹配最多两位小数的金额
业务规则嵌套:数值范围、精度、上下文一致性
语法合法 ≠ 业务合法。例如用户输入年龄“200”,虽能转成int,但明显超出合理范围;再如商品价格“-99.9”,符号错误。
立即学习“Java免费学习笔记(深入)”;
- 在成功解析后,立刻做范围检查:
if (age 150) throw new IllegalArgumentException("年龄应在0~150之间"); - 对金额类字段,强制要求两位小数且非负:
price.scale() == 2 && price.signum() >= 0 - 关联校验:如“结束时间”必须≥“开始时间”,需两个数字解析后比较,不能仅校验单个字符串格式
封装建议:用工具方法统一入口
避免重复写try-catch和校验逻辑,推荐封装静态工具方法:
public static OptionalparseValidInt(String s, int min, int max) {
if (s == null || s.trim().isEmpty()) return Optional.empty();
try {
int val = Integer.parseInt(s.trim());
if (val < min || val > max) return Optional.empty();
return Optional.of(val);
} catch (NumberFormatException e) {
return Optional.empty();
}
}
返回Optional语义清晰,调用方无需判空异常,也便于链式处理。
基本上就这些——合法性检查不是一锤子买卖,而是分层过滤:字符串结构 → 类型转换 → 数值范围 → 业务语义。抓住这四层,就能兼顾健壮性与可维护性。










