如何在 Android 计算器应用中智能限制小数点:仅允许每个数字内出现一次

本文详解如何在 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 Set DELIMITERS = 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 使用,即可优雅解耦业务与校验逻辑。