Java字符串拼接与截取基础教程
在Java开发中,字符串拼接与截取是基础且频繁的操作。本文深入探讨Java字符串处理的核心方法,包括**字符串拼接**与**字符串截取**。对于少量拼接,推荐使用`+`运算符,简单高效;大量拼接则建议选用`StringBuilder`,显著提升性能,多线程环境可选择`StringBuffer`保证线程安全。**字符串截取**主要通过`substring(int beginIndex)`或`substring(int beginIndex, int endIndex)`实现,务必注意索引范围,避免越界异常,并结合`length`、`indexOf`等方法进行安全处理。此外,`String`类还提供`equals`、`contains`、`replace`、`split`等丰富方法,满足比较、查找、替换和分割等多样化文本处理需求。掌握这些基础方法,能有效提升Java字符串处理的效率与安全性。
Java中字符串拼接推荐使用+运算符处理少量拼接,大量拼接时优先选用StringBuilder提升性能,多线程环境下可选StringBuffer保证线程安全;截取主要通过substring(int beginIndex)或substring(int beginIndex, int endIndex)实现,需注意索引范围避免越界,实际开发中应结合length、indexOf等方法进行边界判断与安全处理;此外,String类还提供equals、contains、replace、split等丰富方法,用于比较、查找、替换和分割字符串,满足日常文本处理多样化需求。
Java中对字符串进行拼接和截取是日常开发里最基础也最频繁的操作之一。简单来说,拼接就是把多个字符串连成一个,而截取则是从一个现有字符串中提取出它的一部分。理解并熟练运用这些方法,是处理文本数据的基石。
解决方案
在Java里,字符串的拼接和截取都有多种实现方式,各有特点和适用场景。
字符串拼接
最直观的拼接方式莫过于使用加号 +
运算符。它用起来确实方便,比如:
String part1 = "Hello"; String part2 = "World"; String combined = part1 + " " + part2 + "!"; // 结果: "Hello World!" System.out.println(combined);
这种方式在多数情况下表现良好,尤其是在拼接次数不多的时候。然而,需要注意的是,Java中的 String
对象是不可变的。这意味着每次使用 +
拼接时,都会创建一个新的 String
对象来存储结果,这在循环中进行大量拼接时可能会导致性能问题,因为会产生很多临时的、很快就会被垃圾回收的对象。
为了解决这个问题,Java提供了 StringBuilder
和 StringBuffer
类。它们是可变的字符序列,更适合进行多次字符串修改操作。StringBuilder
非线程安全,性能更高;StringBuffer
线程安全,但性能略低。在单线程环境下,我个人更倾向于使用 StringBuilder
。
// 使用 StringBuilder 进行拼接 StringBuilder sb = new StringBuilder(); sb.append("这是").append("一个").append("StringBuilder").append("的例子。"); String result = sb.toString(); System.out.println(result); // 结果: "这是一个StringBuilder的例子。" // 如果在多线程环境,或者需要线程安全,可以使用 StringBuffer StringBuffer sbuf = new StringBuffer(); sbuf.append("线程安全的").append("拼接。"); String safeResult = sbuf.toString(); System.out.println(safeResult); // 结果: "线程安全的拼接。"
String
类自身也提供了一个 concat()
方法用于拼接,但它和 +
运算符一样,每次调用都会创建新的 String
对象,所以性能上没有本质区别,不建议在循环中大量使用。
String s1 = "Java"; String s2 = "编程"; String s3 = s1.concat(s2); System.out.println(s3); // 结果: "Java编程"
字符串截取
截取字符串主要依赖 String
类的 substring()
方法。它有两个重载版本:
substring(int beginIndex)
: 从指定索引beginIndex
(包含)开始,截取到字符串的末尾。索引是从0开始计数的。String original = "HelloJavaWorld"; String sub1 = original.substring(5); // 从索引5开始截取 System.out.println(sub1); // 结果: "JavaWorld"
substring(int beginIndex, int endIndex)
: 从指定索引beginIndex
(包含)开始,截取到endIndex
(不包含)为止。String original = "HelloJavaWorld"; String sub2 = original.substring(0, 5); // 从索引0到索引5(不包含5) System.out.println(sub2); // 结果: "Hello" String sub3 = original.substring(5, 9); // 从索引5到索引9(不包含9) System.out.println(sub3); // 结果: "Java"
理解索引的“包含”和“不包含”非常关键,尤其是在处理边界情况时。
Java字符串拼接的性能差异及选择考量
在Java中,字符串拼接的性能差异是一个老生常谈的话题,但它确实值得深入理解。这背后主要是 String
对象的不可变性在起作用。当我们用 +
运算符拼接字符串时,比如 String s = "a" + "b" + "c";
,在编译阶段,如果所有操作数都是字面量,Java编译器可能会进行优化,直接生成 String s = "abc";
。但如果操作数中包含变量,例如 String s = str1 + str2 + str3;
,那么在运行时,每次 +
操作都会创建一个新的 String
对象。想象一下在一个循环里拼接几千几万次,这会产生大量的中间 String
对象,给垃圾回收器带来不小的压力,从而影响程序性能。
concat()
方法也面临同样的问题,它的底层实现和 +
运算符在变量拼接时是类似的,每次调用都会返回一个新的 String
对象。
而 StringBuilder
和 StringBuffer
的设计初衷就是为了解决这个问题。它们内部维护一个可变的字符数组,append()
操作不会每次都创建新对象,而是直接在现有数组上进行修改或扩容。只有当你调用 toString()
方法时,才会生成最终的 String
对象。
- 什么时候用
+
或concat()
? 当你需要拼接的字符串数量非常少,比如两三个,或者是在字符串字面量之间拼接时,用+
运算符是最简洁方便的。它的可读性很好,对于这种简单场景,性能开销几乎可以忽略不计。 - 什么时候用
StringBuilder
? 在大多数需要进行多次字符串拼接的场景,尤其是循环内部,或者需要构建复杂字符串时,StringBuilder
是首选。它性能最好,因为它没有同步开销。 - 什么时候用
StringBuffer
? 如果你的应用程序是多线程的,并且多个线程可能同时操作同一个字符串构建器实例,那么StringBuffer
是必要的。它的方法都是同步的,可以保证线程安全,但代价是略微的性能损失。当然,在现代Java开发中,我们更多地会考虑使用ConcurrentHashMap
这样的并发集合,或者通过其他方式避免共享可变状态,但StringBuffer
依然是一个有效的选项。
选择哪种方式,归根结底是根据你的具体需求和对性能、线程安全的要求来决定的。没有绝对的“最好”,只有“最合适”。
Java字符串截取时常见的索引陷阱与边界处理
substring()
方法虽然用起来简单,但如果对索引规则理解不透彻,很容易遇到 IndexOutOfBoundsException
。这是我在实际开发中遇到过不少次的问题,尤其是在处理用户输入或外部系统传来的不确定长度的字符串时。
substring(beginIndex)
和 substring(beginIndex, endIndex)
都要求传入的索引是合法的。所谓合法,就是它们必须在 [0, length]
这个范围内。
常见的陷阱:
beginIndex
小于 0 或大于length()
:String str = "abc"; str.substring(-1);
->IndexOutOfBoundsException
String str = "abc"; str.substring(4);
->IndexOutOfBoundsException
beginIndex
必须在[0, length]
之间。
endIndex
小于beginIndex
或大于length()
:String str = "abc"; str.substring(1, 0);
->IndexOutOfBoundsException
(因为endIndex
小于beginIndex
)String str = "abc"; str.substring(0, 4);
->IndexOutOfBoundsException
(因为endIndex
大于length()
)endIndex
必须在[beginIndex, length]
之间。记住endIndex
是不包含的。
边界处理策略:
预检查字符串长度: 在调用
substring()
之前,先检查字符串是否为空 (null
),或者长度是否符合预期。这是最基本的防御性编程。String text = "Hello"; if (text != null && text.length() >= 5) { String part = text.substring(0, 5); System.out.println(part); } else { System.out.println("字符串不符合截取条件"); }
利用
indexOf()
和lastIndexOf()
查找子串位置: 如果你需要根据某个分隔符或特定子串来截取,结合indexOf()
和lastIndexOf()
是非常实用的。它们返回子串的起始索引,如果找不到则返回-1
,这可以帮助我们避免硬编码索引。String path = "/usr/local/bin/java"; int lastSlash = path.lastIndexOf("/"); if (lastSlash != -1) { String fileName = path.substring(lastSlash + 1); System.out.println("文件名: " + fileName); // 结果: "java" }
处理空字符串或单字符字符串: 对于这些特殊情况,
substring()
依然有效,但你需要确保你的逻辑能正确处理它们。例如,"".substring(0)
会得到空字符串,而"".substring(0,0)
也是空字符串。使用
String.isEmpty()
或String.isBlank()
: 在Java 11及更高版本中,String.isBlank()
可以判断字符串是否为空或只包含空白字符,这比isEmpty()
更全面,有助于提前过滤掉不适合截取的字符串。
总的来说,处理字符串截取时,时刻牢记索引的有效范围,并结合条件判断来规避潜在的运行时错误,是编写健壮代码的关键。
除了拼接与截取,Java字符串还有哪些常用操作方法?
除了拼接和截取,String
类提供了一系列丰富的方法,几乎涵盖了日常文本处理的各种需求。这些方法在数据清洗、格式化、校验等场景中扮演着重要角色。
获取信息类:
length()
: 返回字符串的长度(字符数)。charAt(int index)
: 返回指定索引处的字符。indexOf(String str)
/indexOf(String str, int fromIndex)
: 返回指定子字符串第一次出现的索引。找不到则返回 -1。lastIndexOf(String str)
/lastIndexOf(String str, int fromIndex)
: 返回指定子字符串最后一次出现的索引。
比较与查找类:
equals(Object anObject)
: 比较两个字符串的内容是否相等,区分大小写。equalsIgnoreCase(String anotherString)
: 比较两个字符串的内容是否相等,不区分大小写。contains(CharSequence s)
: 判断字符串是否包含指定的字符序列。startsWith(String prefix)
: 判断字符串是否以指定前缀开头。endsWith(String suffix)
: 判断字符串是否以指定后缀结尾。compareTo(String anotherString)
: 按字典顺序比较两个字符串。
修改与替换类:
replace(char oldChar, char newChar)
: 将字符串中所有出现的oldChar
替换为newChar
。replace(CharSequence target, CharSequence replacement)
: 将所有出现的target
字符序列替换为replacement
。replaceAll(String regex, String replacement)
: 使用正则表达式替换所有匹配项。replaceFirst(String regex, String replacement)
: 使用正则表达式替换第一个匹配项。trim()
: 移除字符串两端的空白字符。strip()
: (Java 11+)更智能地移除两端的空白字符,包括Unicode空白字符。stripLeading()
/stripTrailing()
: (Java 11+)分别移除前导或尾随空白字符。toLowerCase()
: 将字符串转换为小写。toUpperCase()
: 将字符串转换为大写。
分割类:
split(String regex)
: 根据给定的正则表达式将字符串分割成子字符串数组。split(String regex, int limit)
: 限制分割的次数。
格式化与转换类:
valueOf(各种类型)
: 将各种数据类型转换为字符串。这是String.valueOf(123)
这样的静态方法。toCharArray()
: 将字符串转换为字符数组。getBytes()
: 将字符串编码为字节数组。
这些方法共同构成了Java字符串处理的强大工具箱。在实际项目中,我们很少只用到拼接和截取,往往需要组合使用这些方法来完成复杂的文本处理任务,比如解析日志、处理CSV文件、构建SQL查询、验证用户输入格式等等。理解并灵活运用它们,能大大提高开发效率和代码质量。
理论要掌握,实操不能落!以上关于《Java字符串拼接与截取基础教程》的详细介绍,大家都掌握了吧!如果想要继续提升自己的能力,那么就来关注golang学习网公众号吧!

- 上一篇
- Java环境搭建教程,正确配置开发变量方法

- 下一篇
- JavaScript数组引用计数实现解析
-
- 文章 · java教程 | 7分钟前 |
- Java中JSON与XML转换对比分析
- 347浏览 收藏
-
- 文章 · java教程 | 11分钟前 | hashset hashcode() equals() JavaSet 元素查找效率
- JavaSet元素判断方法详解
- 273浏览 收藏
-
- 文章 · java教程 | 13分钟前 |
- 工厂模式三种实现方法对比解析
- 192浏览 收藏
-
- 文章 · java教程 | 23分钟前 |
- Java文件复制方法详解:流与Files.copy对比
- 112浏览 收藏
-
- 文章 · java教程 | 1小时前 |
- 非200响应处理方法及日志记录技巧
- 146浏览 收藏
-
- 文章 · java教程 | 1小时前 |
- Java记录类与不可变对象使用技巧
- 100浏览 收藏
-
- 文章 · java教程 | 1小时前 |
- Java实现PBKDF2密码验证教程
- 163浏览 收藏
-
- 文章 · java教程 | 1小时前 |
- AndroidImageView锚点缩放技巧详解
- 493浏览 收藏
-
- 文章 · java教程 | 2小时前 |
- JavaIO/NIO原理与优化技巧详解
- 381浏览 收藏
-
- 文章 · java教程 | 2小时前 |
- Redis分布式锁优化与问题解决全攻略
- 355浏览 收藏
-
- 文章 · java教程 | 3小时前 |
- Java压缩文件教程:ZipOutputStream使用详解
- 247浏览 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 542次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 511次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 498次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 484次学习
-
- 千音漫语
- 千音漫语,北京熠声科技倾力打造的智能声音创作助手,提供AI配音、音视频翻译、语音识别、声音克隆等强大功能,助力有声书制作、视频创作、教育培训等领域,官网:https://qianyin123.com
- 191次使用
-
- MiniWork
- MiniWork是一款智能高效的AI工具平台,专为提升工作与学习效率而设计。整合文本处理、图像生成、营销策划及运营管理等多元AI工具,提供精准智能解决方案,让复杂工作简单高效。
- 191次使用
-
- NoCode
- NoCode (nocode.cn)是领先的无代码开发平台,通过拖放、AI对话等简单操作,助您快速创建各类应用、网站与管理系统。无需编程知识,轻松实现个人生活、商业经营、企业管理多场景需求,大幅降低开发门槛,高效低成本。
- 190次使用
-
- 达医智影
- 达医智影,阿里巴巴达摩院医疗AI创新力作。全球率先利用平扫CT实现“一扫多筛”,仅一次CT扫描即可高效识别多种癌症、急症及慢病,为疾病早期发现提供智能、精准的AI影像早筛解决方案。
- 196次使用
-
- 智慧芽Eureka
- 智慧芽Eureka,专为技术创新打造的AI Agent平台。深度理解专利、研发、生物医药、材料、科创等复杂场景,通过专家级AI Agent精准执行任务,智能化工作流解放70%生产力,让您专注核心创新。
- 212次使用
-
- 提升Java功能开发效率的有力工具:微服务架构
- 2023-10-06 501浏览
-
- 掌握Java海康SDK二次开发的必备技巧
- 2023-10-01 501浏览
-
- 如何使用java实现桶排序算法
- 2023-10-03 501浏览
-
- Java开发实战经验:如何优化开发逻辑
- 2023-10-31 501浏览
-
- 如何使用Java中的Math.max()方法比较两个数的大小?
- 2023-11-18 501浏览