C++20起std::string::ends_with可直接使用,支持string_view、const char*或char;C++17及更早需手动实现,推荐用compare避免substr开销,并注意无符号下溢、宽字符类型匹配及UTF-8多字节问题。
如果你的编译器支持 C++20(如 GCC 10+、Clang 10+、MSVC 19.30+),std::string 和 std::string_view 已原生提供 ends_with() 成员函数,无需手动实现。
它接受 std::string_view、const char* 或单个 char,返回 bool:
std::string s = "example.txt";
if (s.ends_with(".txt")) {
// true
}注意:该函数区分大小写,且不进行路径规范化 —— "file.TXT" 对 ".txt" 返回 false。
核心思路是比对尾部子串:先检查长度是否足够,再用 compare() 或 substr() 判断后缀是否匹配。推荐用 compare(),避免临时 std::string 构造开销。
compare() 的高效写法:
bool ends_with(const std::string& s, const std::string& suffix) {
if (suffix.length() > s.length()) return false;
return s.compare(s.length() - suffix.length(), suffix.length(), suffix) == 0;
}常见误用点:
s.length() - suffix.length() 下溢为极大正数(size_t 无符号)s.substr(...).compare(suffix),触发不必要的内存分配== 比较 substr() 结果,同样有构造开销std::wstring 同样适用上述逻辑,但需确保后缀参数类型一致:
std::wstring ws = L"数据.csv";
if (ws.size() >= 4 && ws.compare(ws.size() - 4, 4, L".csv") == 0) {
// 注意:L".csv" 是 wchar_t 字符串字面量
}关键点:
std::string 和 std::wstring 的 compare() 不互通,不能混用std::string 存储 UTF-8)不能靠字节长度判断后缀 —— 多字节字符会使 .length() ≠ 字符数;此时需用 ICU 或 std::codecvt_utf8(已弃用)等库做 Unicode 感知切分,但绝大多数文件扩展名场景无需考虑在高频调用(如日志过滤、路径扫描)中,以下细节影响实际表现:
".log")时,可改用模板推导长度,避免运行时 strlen():compare() 优化为内联字节比较"")始终返回 true,符合语义直觉,但需确认业务逻辑是否允许最易被忽略的是无符号整数下溢 —— 即便加了长度判断,若写成 s.length() >= suffix.length() && s.compare(...),中间仍可能因短路求值失败而跳过检查;稳妥做法是显式分支或使用 static_cast 转换(需包

)。