php中应使用substr配合strpos(或strrpos)截取指定字符后的子串,需检查strpos返回值是否为false,并注意utf-8多字节字符需用mb_strpos/mb_substr。

用 substr + strpos 截取指定字符后的子串
PHP 里没有叫 piso 的函数,你大概率是记混了名字,实际想用的是 substr 配合 strpos(或 strrpos)来定位并截取某字符之后/之前的内容。
典型场景:从 "user_12345@domain.com" 中提取 "12345"(下划线后、@号前);或从路径 "/api/v2/users/1001" 中取末尾的 ID。
-
strpos($str, $needle)返回第一次出现的位置(从 0 开始),没找到返回false -
strrpos($str, $needle)返回最后一次出现的位置,适合处理可变长度前缀(比如文件扩展名) -
substr($str, $start, $length)的$start可为负数,但用strpos算出的正数偏移更直观
截取「某个字符之后」的全部内容(含边界判断)
错误写法:substr($str, strpos($str, '_')) —— 这会保留下划线本身。正确做法是把起始位置 +1。
安全写法必须检查 strpos 是否返回 false,否则 substr 会从字符串末尾开始截(PHP 将 false 转为 0,但 false === 0 是 false,直接用会触发 notice):
立即学习“PHP免费学习笔记(深入)”;
if (($pos = strpos($str, '_')) !== false) {
$id = substr($str, $pos + 1);
}如果还要再切掉后面的 @,可以链式用两次:substr($str, $pos + 1, strpos($str, '@', $pos) - $pos - 1),注意第三个参数是长度,不是结束位置。
截取「两个字符之间」的内容(如中间字段)
这是最容易出错的地方:用 strpos 找第一个分隔符,再用它找第二个时,必须传入第三个参数(起始搜索位置),否则第二次 strpos 还是从头找,结果可能错乱。
- 错误:
$end = strpos($str, '@')→ 如果$str = "a@b@c",会得到第一个@的位置 - 正确:
$end = strpos($str, '@', $start),其中$start = strpos($str, '_') + 1 - 最终截取:
substr($str, $start, $end - $start)
示例:从 "order_no:ORD-2024-001|status:paid" 中取 "ORD-2024-001":
$start = strpos($str, ':') + 1; $end = strpos($str, '|', $start); $result = $end !== false ? substr($str, $start, $end - $start) : substr($str, $start);
为什么不用 explode?什么时候该换
explode 看似简单,但遇到分隔符重复、空字段、或只需要第 N 段时,容易越界或逻辑膨胀。比如 explode('@', $email)[0] 取邮箱用户名是对的,但 explode('/', $path)[3] 在路径深度不固定时就不可靠。
真正需要「精确位置控制」时,strpos + substr 更稳——尤其是处理日志行、协议头、或自定义格式字符串。不过要注意:它们只支持 byte 级别查找,对 UTF-8 多字节字符(如中文、emoji)会出错,此时得换 mb_strpos 和 mb_substr。
最常被忽略的一点:所有 strpos 系函数返回的是字节偏移,不是字符索引。中文字符串里直接套用会切出乱码,这点在接口解析用户昵称或地址时特别致命。











