如何通过 String.substring() 方法从指定符号后截取字符串的子串?
- 内容介绍
- 相关推荐
本文共计918个文字,预计阅读时间需要4分钟。
javaString substring(String str, int start, int end) { int index=str.indexOf(''); // 找到指定字符的位置 if (index==-1) { return ; // 如果没有找到字符,返回空字符串 } return str.substring(index + 1, end); // 从指定字符后截取字符串}
常见错误是直接写 str.substring("@") —— 这会编译失败,因为 substring() 没有接收字符串的重载方法。
- 如果符号可能多次出现,用
indexOf()得到第一次位置,lastIndexOf()得到最后一次位置 - 务必检查返回值:找不到时
indexOf()返回-1,直接传给substring()会抛StringIndexOutOfBoundsException - 截取“符号之后”的内容,起始索引通常是
pos + 1(跳过符号本身)
正确写法:先找位置,再截取,且必须判空和容错
以提取邮箱中 @ 后的域名部分为例:
String email = "user@example.com"; int atPos = email.indexOf('@'); String domain = (atPos == -1) ? "" : email.substring(atPos + 1);
这个逻辑看似简单,但容易漏掉边界情况:
-
email为null→ 调用indexOf()前需判空,或用Objects.requireNonNull() -
email是空字符串""→indexOf()返回-1,没问题,但得确保后续逻辑能处理空结果 - 符号在末尾,如
"abc@"→atPos + 1等于字符串长度,substring(len)是合法的,返回空字符串"",不是异常
遇到多个相同符号时,lastIndexOf() 更符合直觉
比如路径 /home/user/file.txt,想取扩展名(最后一个 . 之后的部分),用 indexOf() 会错拿 /home/user/file 中第一个 . 的位置。
应改用:
String path = "/home/user/file.txt"; int lastDot = path.lastIndexOf('.'); String ext = (lastDot == -1) ? "" : path.substring(lastDot + 1); // "txt"
注意:lastIndexOf() 同样返回 -1 表示未找到,不能省略判断;且它只搜字符串,不支持正则。
- 若要支持更复杂的分隔逻辑(如忽略引号内的点),
substring()+indexOf()就不够用了,得换split()或正则 -
lastIndexOf(char)比lastIndexOf(String)性能略好,单字符分隔优先用前者
替代方案:split() 更简洁,但要注意空结尾和性能
对简单分隔,split() 写起来更短:
String domain = email.split("@", 2).length > 1 ? email.split("@", 2)[1] : "";
但问题明显:
- 调用了两次
split(),低效;应存中间结果 -
split()遇到结尾分隔符(如"a@b@")默认丢弃空字段,需加负数参数:split("@", -1) - 正则特殊字符(
.、*、?等)必须转义,而indexOf()不需要
真正需要灵活切分时,别硬套 substring();但只要目标是“某符号之后全部内容”,indexOf() + substring() 仍是最快最可控的方式。
最容易被忽略的是:符号位置计算后没做越界检查,尤其当输入来自外部(表单、文件、网络)时,-1 和 length 边界必须显式处理,否则线上崩得无声无息。
本文共计918个文字,预计阅读时间需要4分钟。
javaString substring(String str, int start, int end) { int index=str.indexOf(''); // 找到指定字符的位置 if (index==-1) { return ; // 如果没有找到字符,返回空字符串 } return str.substring(index + 1, end); // 从指定字符后截取字符串}
常见错误是直接写 str.substring("@") —— 这会编译失败,因为 substring() 没有接收字符串的重载方法。
- 如果符号可能多次出现,用
indexOf()得到第一次位置,lastIndexOf()得到最后一次位置 - 务必检查返回值:找不到时
indexOf()返回-1,直接传给substring()会抛StringIndexOutOfBoundsException - 截取“符号之后”的内容,起始索引通常是
pos + 1(跳过符号本身)
正确写法:先找位置,再截取,且必须判空和容错
以提取邮箱中 @ 后的域名部分为例:
String email = "user@example.com"; int atPos = email.indexOf('@'); String domain = (atPos == -1) ? "" : email.substring(atPos + 1);
这个逻辑看似简单,但容易漏掉边界情况:
-
email为null→ 调用indexOf()前需判空,或用Objects.requireNonNull() -
email是空字符串""→indexOf()返回-1,没问题,但得确保后续逻辑能处理空结果 - 符号在末尾,如
"abc@"→atPos + 1等于字符串长度,substring(len)是合法的,返回空字符串"",不是异常
遇到多个相同符号时,lastIndexOf() 更符合直觉
比如路径 /home/user/file.txt,想取扩展名(最后一个 . 之后的部分),用 indexOf() 会错拿 /home/user/file 中第一个 . 的位置。
应改用:
String path = "/home/user/file.txt"; int lastDot = path.lastIndexOf('.'); String ext = (lastDot == -1) ? "" : path.substring(lastDot + 1); // "txt"
注意:lastIndexOf() 同样返回 -1 表示未找到,不能省略判断;且它只搜字符串,不支持正则。
- 若要支持更复杂的分隔逻辑(如忽略引号内的点),
substring()+indexOf()就不够用了,得换split()或正则 -
lastIndexOf(char)比lastIndexOf(String)性能略好,单字符分隔优先用前者
替代方案:split() 更简洁,但要注意空结尾和性能
对简单分隔,split() 写起来更短:
String domain = email.split("@", 2).length > 1 ? email.split("@", 2)[1] : "";
但问题明显:
- 调用了两次
split(),低效;应存中间结果 -
split()遇到结尾分隔符(如"a@b@")默认丢弃空字段,需加负数参数:split("@", -1) - 正则特殊字符(
.、*、?等)必须转义,而indexOf()不需要
真正需要灵活切分时,别硬套 substring();但只要目标是“某符号之后全部内容”,indexOf() + substring() 仍是最快最可控的方式。
最容易被忽略的是:符号位置计算后没做越界检查,尤其当输入来自外部(表单、文件、网络)时,-1 和 length 边界必须显式处理,否则线上崩得无声无息。

