C++如何判断一个字符串是否为数字?(代码示例)

c++kquote>推荐使用std::stod配合位置检查判断字符串是否为合法数字;若需严格格式控制则手动遍历;正则适用于复杂多变规则场景。

直接用标准库的 std::stoistd::stod 等函数配合异常处理,是最常用且可靠的方式;若需严格校验(比如不允许前后空格、不允许科学计数法),则建议手动遍历或用正则。

方法一:用 std::stod + 异常捕获(推荐,兼顾简洁与健壮)

适用于判断是否为合法的浮点数(也兼容整数),能自动跳过开头空格,但要求整个字符串必须是有效数字(末尾不能有多余字符)。

关键点:调用后需检查是否“消费了全部字符”,否则像 "123abc" 会被误判为合法。

  • 先用 std::stod(str, &pos) 尝试转换,获取解析结束位置 pos
  • 再判断 pos == str.length()str 非空
  • 同时捕获 std::invalid_argument(格式错误)和 std::out_of_range(溢出)
#include 
#include 
#include 

bool isNumber(const std::string& s) {
    if (s.empty()) return false;
    size_t pos = 0;
    try {
        std::stod(s, &pos);
        return pos == s.length();
    } catch (const std::invalid_argument&) {
        return false;
    } catch (const std::out_of_range&) {
        return false;
    }
}

// 测试
int main() {
    std::cout << isNumber("123") << "\n";      // 1
    std::cout << isNumber("-45.67") << "\n";   // 1
    std::cout << isNumber(" 123 ") << "\n";    // 0(含空格→失败)
    std::cout << isNumber("123abc") << "\n";   // 0
    std::cout << isNumber("") << "\n";         // 0
}

方法二:手动遍历(完全可控,适合自定义规则)

适合需要严格限制格式的场景,例如:只允许整数、禁止小数点、禁止正号、禁止空格等。

逻辑清晰,无异常开销,但需自己处理符号、小数点、指数等细节。

  • 允许开头有可选的 '+' 或 '-'
  • 必须至少有一个数字;小数点最多一个,且不能在开头或结尾单独出现
  • 不支持 'e'/'E' 科学计数法(如需支持,需额外扩展)
bool isSimpleNumber(const std::string& s) {
    if (s.empty()) return false;
    size_t i = 0;

    // 检查符号
    if (s[i] == '+' || s[i] == '-') i++;
    if (i == s.length()) return false; // 只有符号

    bool hasDot = false;
    bool hasDigit = false;

    for (; i < s.length(); ++i) {
        char c = s[i];
        if (c == '.') {
            if (hasDot) return false; // 多个小数点
            hasDot = true;
        } else if (c >= '0' && c <= '9') {
            hasDigit = true;
        } else {
            return false; // 非数字非小数点
        }
    }

    return hasDigit; // 至少一位数字
}

方法三:用正则表达式(C++11 起支持,适合复杂模式)

适合规则明确且可能变化的场景,比如要支持整数、浮点、科学计数法、带千分位符等。

注意:正则性能略低,且不同编译器对 std::regex 支持度不一(MSVC 较好,旧版 libstdc++ 可能慢或不全)。

#include 

bool isNumberRegex(const std::string& s) {
    // 匹配:可选符号 + 数字 + 可选小数部分 + 可选指数部分
    const std::regex pattern(R"(^[+-]?(\d+\.?\d*|\.\d+)([eE][+-]?\d+)?$)");
    return !s.empty() && std::regex_match(s, pattern);
}

基本上就这些。日常开发中,方法一足够用;对性能敏感或规则特殊时,选方法二;需要灵活匹配变体时,再考虑方法三。