c++如何判断字符串相等_c++ string比较函数

使用 == 比较 std::string 最安全、简洁、类型安全,等价于长度与内容逐字符比对;误用 strcmp 易出错,compare() 适用于三路比较而非单纯判等。

直接用 == 比较 std::string 是最安全的做法

不需要调用额外函数,std::string 重载了 == 运算符,语义清晰、类型安全、自动处理长度和内容比对。它等价于逐字符比较并检查长度是否一致,底层调用的是 std::char_traits::compare,但你完全不用关心这个。

常见错误是误用 C 风格的 strcmp:它只接受 const char*,对空字符串或未以 \0 结尾的 std::string 数据(比如含嵌入 null)会出错或截断;还容易忘记调用 .c_str() 导致编译失败。

  • std::string a = "hello"; std::string b = "hello"; if (a == b) { ... } ✅ 正确、简洁、可读性强
  • if (strcmp(a.c_str(), b.c_str()) == 0) ⚠️ 多余、不安全(如 a\0)、易漏 .c_str()
  • if (a.compare(b) == 0) ✅ 功能正确,但不如 == 直观,且多写 6 个字符

compare() 的用途不是判断相等,而是做三路比较

std::string::compare() 返回 int:负值表示小于,0 表示等于,正值表示大于。它主要用于排序、二分查找或需要知道大小关系的场景,比如传给 std::sort 的自定义谓词,或实现字典序逻辑。

如果只是判断是否相等,用 compare() 属于“杀鸡用牛刀”,且容易写成 if (a.compare(b))(把“不等”当真值),引发逻辑翻转 bug。

  • 用于排序:
    std::vector v = {"zebra", "apple"}; std::sort(v.begin(), v.end(), [](const auto& x, const auto& y) { return x.compare(y) < 0; });
  • 错误写法:if (a.compare(b)) { /* 这里执行的是 a != b 时的分支 */ }
  • 正确等价写法:if (a.compare(b) == 0),但还是推荐直接写 a == b

区分大小写的比较就用 ==,忽略大小写需手动转换或使用 locale

std::string 默认所有操作(包括 ==compare)都是区分大小写的。C++ 标准库没有内置的忽略大小写比较函数,不要试图用 std::toupper 在循环中逐字符比较——它依赖当前 locale,且对非 ASCII 字符(如 `é`, `ß`)行为不可靠。

  • 简单 ASCII 场景(仅 a–z/A–Z):
    bool iequals(const std::string& a, const std::string& b) {
        if (a.length() != b.length()) return false;
        for (size_t i = 0; i < a.length(); ++i)
            if (std::tolower(static_cast(a[i])) != 
                std::tolower(static_cast(b[i])))
                return false;
        return true;
    }
  • 真实项目建议用 ICU 或 Boost.Locale,避免自己实现 Unicode 大小写折叠
  • 别用 std::equal(a.begin(), a.end(), b.begin(), [](char x, char y) { return tolower(x) == tolower(y); }) —— 同样有 locale 和 unsigned char 问题

注意 std::string_view 的比较行为与 std::string 一致

C++17 引入的 std::string_view 也重载了 ==,并且能和 std::string、C 字符串字面量自然比较,无需转换:

  • std::string_view sv = "test"; std::string s = "test"; bool eq = (sv == s);
  • eq = (sv == "test"); ✅ 字面量隐式转为 string_view
  • sv == nullptr 是未定义行为,传入空指针前必须确保非空
  • string_view 指向的数据生命周期短于比较操作,会导致悬垂视图 —— 这是唯一比 std::string 更容易出错的地方

实际项目里,95% 的字符串相等判断用 == 就够了。真正容易被忽略的是:比较前没确认对象是否有效(比如未初始化的 string_view)、在跨编码或国际化场景下硬写大小写转换、以及把 compare() 当布尔表达式直接用。