substr是std::string成员函数,非全局函数,不可用于char*;pos为起始下标(≤size()),len为长度(非结束下标),越界自动截断,负数传参会因size_t转换导致异常。
直接说结论:substr 是 C++ std::string 的成员函数,**不是全局函数,不能对 C 风格字符串(char*)直接调用**;它默认从指定位置开始、向后截取指定长度,越界时会自动截断到末尾,不会崩溃但可能不符合预期。
substr 原型是 string substr(size_t pos = 0, size_t len = npos) const,其中:
pos 是起始下标(从 0 开始),必须 ;若等于 size(),返回空串;若大于 size(),抛出 std::out_of_range
len 是要截取的字符数,不是结束下标;若 len 超出剩余字符数,实际只取到末尾npos 是一个特殊值(通常是 size_t(-1)),表示“直到结尾”,不是“无限长”——它只是让函数内部按 size() - pos 计算比如想从第 3 个字符取到倒数第 2 个,写成 s.substr(3, s.size()-2) 是错的——这会从下标 3 开始取 s.size()-2 个字符,远超实际范围。
正确做法分两步判断:
size_t start = 3;
size_t len = (s.size() >= 5) ? s.size() - 2 - start : 0;(避免无符号整数下溢)s.substr(start, len)
以下写法全部非法:
const char* cstr = "hello world"; cstr.substr(0, 5); // ❌ 编译失败:char* 没有 substr 成员 std::substr(cstr, 0, 5); // ❌ 没有这个全局函数
必须显式构造:
std::string(cstr).substr(0, 5)(临时对象,适合短逻辑)std::string s(cstr); auto sub = s.substr(0, 5);(适合多次操作)std::string_view(C++17),它也有 substr,行为一致但不分配内存substr 返回的是新 std::string,内部会分配内存并复制字符。对几 MB 的字符串反复调用,容易触发频繁堆分配。
替代方案视场景而定:
std::string_view + subst
r(零拷贝)std::string,但尽量复用变量,避免链式调用如 s.substr(a,b).substr(c,d)
std::vector<:string_view> 管理片段,最后统一拼接最常被忽略的一点:pos 是 size_t,传入负数会因类型转换变成极大正数,直接越界抛异常——别用 int 变量直接传参。