
本文详解如何在 android 计算器中实现「按数字粒度校验小数点」——即允许多个数字各自含一个`.`(如 `3.14+2.5*0.007`),但禁止同一数字内重复输入小数点(如 `2..5` 或 `10.5.3`),避免全局单点限制导致的功能僵化。
在开发 Android 计算器应用时,一个常见却易被误处理的交互逻辑是:小数点 . 的输入控制。很多开发者采用“全局计数”方式(例如用布尔变量 hasDecimal = true/false),结果导致用户无法输入类似 5.2 + 0.75 - 100.001 这样的合法表达式——因为第二个数字前的小数点被错误拦截。
正确方案是:以“当前操作数(operand)”为单位进行小数点校验。也就是说,每当用户输入一个运算符(+, -, *, /, %, (, ) 等)或括号时,就标志着前一个数字已结束、新数字即将开始,此时应重置该数字内的小数点状态。
✅ 核心思路:动态追踪“当前数字起始位置”
我们维护一个 rightmostDelimiterIdx 变量,记录最近一个分隔符字符(如 +, -, *, /, (, ) 等)在输入字符串中的索引。那么,从 rightmostDelimiterIdx + 1 到当前字符串末尾,就是当前正在输入的数字部分。只需检查该子串中是否已存在 .,即可安全决定是否追加新的小数点。
以下是精简可复用的核心逻辑(适配 Android StringBuilder 输入管理):
private int rightmostDelimiterIdx = 0; private final SetDELIMITERS = Set.of('+', '-', '*', '/', '%', '(', ')', '[', ']'); private void onCharTyped(StringBuilder input, char c) { if (DELIMITERS.contains(c)) { // 遇到运算符或括号:标记新数字开始位置 rightmostDelimiterIdx = input.length(); input.append(c); } else if (c == '.') { // 检查当前数字(从上一个分隔符后到末尾)是否已有小数点 int startOfCurrentNumber = rightmostDelimiterIdx; if (input.indexOf(".", startOfCurrentNumber) == -1) { input.append('.'); } // 否则忽略重复小数点(静默丢弃,不报错) } else if (Character.isDigit(c) || c == 'e' || c == 'E') { // 允许数字、科学计数法符号等 input.append(c); } // 其他字符(如空格、非法符号)可根据需求过滤 }
⚠️ 注意事项与进阶建议
- 删除逻辑需同步更新:当用户点击退格(Backspace)时,若删掉的是分隔符(如 +),需向前搜索上一个分隔符重新设置 rightmostDelimiterIdx;若删掉的是小数点,无需特殊处理(因校验是实时的)。
- 支持负数:若允许 -5.2,需将 - 视为数字前缀而非分隔符——建议扩展判断逻辑:仅当 - 出现在表达式开头 或 紧跟在 DELIMITERS 后时,才不视为分隔符(即 5+-3 中的 - 是分隔符,而 5+(-3) 或 -2.5 中的 - 不是)。
- 输入框 UI 提示:可在 EditText 的 TextWatcher 中调用上述逻辑,并结合 setError() 或图标反馈,提升用户体验。
- 避免 JS 引擎依赖:原文示例使用 ScriptEngine 仅用于演示表达式求值,Android 开发中应避免在主线程使用 ScriptEngine(性能差、API 26+ 已弃用)。推荐改用 exp4j 或手写简易表达式解析器。
✅ 总结
实现“每数字限一个点”的关键,在于放弃全局状态,转而基于语法结构(即运算符/括号划分的操作数边界)做局部校验。这不仅解决了多小数点输入问题,也为后续支持科学计数法、负数、函数调用(如 sin(3.14))等复杂表达式打下坚实基础。在 Android 中,将该逻辑封装为 InputValidator 类,配合 TextWatcher 使用,即可优雅解耦业务与校验逻辑。









