用 string::find 判断子串存在需检查返回值是否等于 string::npos,因它返回 size_t 类型,与 -1 比较会出错;支持指定起始位置搜索,但区分大小写且仅字节级匹配,空串总返回 0。

用 string::find 判断子串是否存在最直接
直接调用 find,检查返回值是否等于 string::npos 就行。它返回子串首次出现的起始下标(size_t 类型),没找到才返回 string::npos —— 这个值保证大于任何合法下标,所以不能用 == -1 判断(-1 会被转成极大正数,比较永远为假)。
常见错误写法:if (s.find("abc") != -1) —— 这是错的,string::npos 是无符号类型,和 -1 比较会出问题。
正确写法示例:
string s = "hello world";
if (s.find("world") != string::npos) {
// 找到了
}
find 的重载版本和搜索起点控制
默认从下标 0 开始找,但你可以指定起始位置,避免重复扫描或实现“第 N 次出现”的逻辑。
立即学习“C++免费学习笔记(深入)”;
-
s.find(sub):从开头找 -
s.find(sub, pos):从下标pos开始向后找(pos超出长度时自动返回npos) -
find_first_of和find_last_of不是找子串,而是找任意一个字符匹配(慎用,容易误判)
注意:pos 是 size_t,传负数会整数溢出,务必确保非负。
区分大小写的限制与替代方案
string::find 是严格区分大小写的,比如 "Hello".find("hello") 返回 npos。
没有内置不区分大小写的查找,得自己处理:
- 临时转小写再查(适合短字符串,注意 locale 影响)
- 手写循环逐字符比较(可控、无额外分配)
- 用
std::search配合自定义谓词(C++17 起更简洁)
简单转小写示例(仅 ASCII 安全):
string lower_s = s;
transform(lower_s.begin(), lower_s.end(), lower_s.begin(), ::tolower);
if (lower_s.find("hello") != string::npos) { ... }
性能和边界情况要注意什么
find 平均时间复杂度是 O(n×m),最坏情况(如全是相同字符)接近 O(n×m),不是 KMP 级别优化。
几个易忽略点:
- 空子串
""总是被找到,且返回0(C++ 标准规定) - 子串比原串长,直接返回
npos,不崩溃 - 多字节字符(如 UTF-8)会被当作字节序列处理,
find不识别 Unicode 码点 - 频繁查找建议预处理成
std::string_view(C++17),避免重复构造
真正要高效、安全地处理 Unicode 子串匹配,得换用 ICU 或 Boost.Locale,find 只负责字节级精确匹配。











