substring的两个参数是索引位置,beginIndex为起始保留位置(含),endIndex为首个不保留位置(不含),区间为左闭右开。

substring 的两个参数到底指什么位置
substring 不是“截取第几个到第几个字符”,而是按**索引位置**切片,且起始包含、结束不包含。Java 字符串索引从 0 开始,substring(int beginIndex, int endIndex) 中:beginIndex 是第一个要保留字符的位置,endIndex 是第一个**不保留**字符的位置。
常见错误现象:"hello".substring(1, 3) 返回 "el"(不是 "ell" 或 "ll"),因为索引 1 是 'e',索引 3 是 'l' 后面那个位置(即第 4 个字符前),所以取 [1, 3) 区间。
- 如果
beginIndex == endIndex,结果是空字符串"" - 如果
beginIndex > endIndex,抛出StringIndexOutOfBoundsException -
endIndex可以等于字符串长度(s.length()),这是合法的,表示“截到末尾”
只传一个参数时的默认行为
substring(int beginIndex) 是重载方法,它等价于 substring(beginIndex, s.length()) —— 从指定位置一直截到末尾。
使用场景:提取后缀、去掉前缀(如去掉路径前的固定目录)、日志中截取时间戳之后的内容。
立即学习“Java免费学习笔记(深入)”;
-
"2024-05-20T14:30:00Z".substring(11)→"14:30:00Z" - 但要注意:如果
beginIndex超出范围(比如大于s.length()),直接抛异常;等于s.length()则返回空串 - 没有“负数索引”支持,不像 Python,想从末尾算需手动计算:
s.substring(s.length() - 3)
容易踩的坑:空指针和越界
调用 substring 前,字符串本身是否为 null 比索引错误更早炸——null.substring(...) 直接抛 NullPointerException。
而索引越界分两种:beginIndex 或 <code>endIndex > s.length() 都会触发 StringIndexOutOfBoundsException;但 beginIndex > s.length() 也会报错(哪怕 endIndex 更大)。
- 别假设输入一定非空,先判
if (s != null && !s.isEmpty()) - 如果索引来自用户输入或外部解析(如正则匹配的
start()/end()),务必校验范围:Math.max(0, Math.min(s.length(), idx))不推荐——它可能掩盖逻辑错误,应显式处理边界情况 - 注意 substring 在 Java 7u6 之后已不共享底层 char[],内存安全了,但短字符串反复截取仍有小对象开销
替代方案:用 String.indexOf + substring 安全定位
硬写数字索引容易错,尤其当目标子串位置不固定时。更健壮的做法是先用 indexOf 找位置,再传给 substring。
例如提取 URL 中 ?token= 后的值:
String url = "https://api.example.com/login?token=abc123&expire=3600";
int start = url.indexOf("token=");
if (start != -1) {
int end = url.indexOf('&', start);
String token = url.substring(start + 6, end == -1 ? url.length() : end);
}关键点:
-
indexOf找不到返回-1,必须检查,否则substring(-1, ...)必炸 -
start + 6是跳过"token="的长度,硬编码需确保字符串字面量没写错 - 第二个
indexOf查找分隔符,避免把&expire也吞进去
边界情况多的时候,别省那几行校验——线上 StringIndexOutOfBoundsException 往往就卡在少了一个 != -1 判断。










