Java8 对字符串连接的改进

旧城等待, 2021-09-27 10:58 389阅读 0赞

转自:Java8 对字符串连接的改进

我们提出一个需求:有一个 List<String>,将其格式化为 元素1, 元素2, 元素3, … 元素N 的字符串形式

毋庸置疑,Java8 之前我们的第一反应是使用 StringBuilder

  1. public static String formatList(List<String> list, String delimiter) {
  2. StringBuilder result = new StringBuilder();
  3. for (String str : list) {
  4. result.append(str).append(delimiter);
  5. }
  6. // 删除末尾多余的 delimiter
  7. result.delete(result.length() - delimiter.length(), result.length());
  8. return result.toString();
  9. }
  10. public static void main(String[] args) throws Exception {
  11. List<String> list = Arrays.asList("a", "b", "c", "d", "e", "f", "g");
  12. System.out.println("使用 StringBuilder:");
  13. String format = formatList(list, ", ");
  14. System.out.println(format);
  15. }

运行结果:

  1. 使用 StringBuilder
  2. formatList1:a, b, c, d, e, f, g

JDK1.8 时,添加了一个新的用于字符串连接的类,专门用于这种需要 分隔符 的场合,它就是 StringJoinerStringJoiner 在构造时可以指定一个分隔符(delimiter),然后每连接一个元素它便会加上一个 delimiter,使用 StringJoiner 改写 formatList

  1. public static String formatList(List<String> list, String delimiter) {
  2. StringJoiner result = new StringJoiner(delimiter);
  3. for (String str : list) {
  4. result.add(str);
  5. }
  6. return result.toString();
  7. }
  8. public static void main(String[] args) throws Exception {
  9. List<String> list = Arrays.asList("a", "b", "c", "d", "e", "f", "g");
  10. System.out.println("使用 StringJoiner:");
  11. String format = formatList(list, ", ");
  12. System.out.println(format);
  13. }

运行结果:

  1. 使用 StringJoiner
  2. formatList2:a, b, c, d, e, f, g

formatList 的代码更加的简洁了 —— 当然我们还可以更简洁 —— JDK1.8 为 String 类添加了一个新的静态方法,String.join

2307344969-58c6b0f0c0377

可以看到,第一个参数为一个分隔符 delimiter,第二个参数可以是一个 Iterable,或者是一个变长参数的 CharSequenceString 就是 CharSequence 的子类)。
所以 formatList 方法只需要一句代码就搞定:

  1. public static String formatList(List<String> list, String delimiter) {
  2. return String.join(delimiter, list);
  3. }

当然,我们能猜到,String.join 方法的功能内部是通过 StringJoiner 来实现的,String.join (CharSequence, Iterable) 的源码:

80707697-5854ea530288b


但是我们看到了 String.join 方法的不足 —— 它不能指定前缀和后缀 —— 比如我们如果想要直接将 List<String> 格式化为 { 元素1, 元素2, 元素3, … 元素N } 呢?(此时前缀为 "{ ",后缀为 " }"

查看 StringJoiner 的构造方法,发现 StringJoiner 除了指定 分隔符 的构造方法,还有一个可以指定 分隔符、前缀和后缀 的构造方法:

1339755913-58c6b112aa18a

修改 formatList

  1. public static String formatList(
  2. List<String> list, String delimiter, String prefix, String suffix) {
  3. StringJoiner result = new StringJoiner(delimiter, prefix, suffix);
  4. for (String str : list) {
  5. result.add(str);
  6. }
  7. return result.toString();
  8. }
  9. public static void main(String[] args) throws Exception {
  10. List<String> list = Arrays.asList("a", "b", "c", "d", "e", "f", "g");
  11. System.out.println("使用 StringJoiner,带前缀和后缀:");
  12. String format = formatList(list, ", ", "{ ", " }");
  13. System.out.println(format);
  14. }

运行结果:


很棒~ 但能不能更简洁呢? 事实上,Java8 对于字符串集合的连接操作提供了一个专门的流式 API,即 Collectors.joining 函数:
Collectors.joining 函数

  • 无参的 joining() 方法,即不存在连接符(底层实现为 StringBuilder);
  • joining(CharSequence delimiter) 方法,即分隔符为 delimiter(底层实现为 StringJoiner);
  • joining(CharSequence delimiter, CharSequence prefix, CharSequence suffix)方法,即分隔符为 delimiter,前缀为 prefix,后缀为 suffix(底层实现为 StringJoiner)。

那怎么使用呢? 我们直接使用三个参数的 Collectors.joining 方法改写 formatList

  1. public static String formatList(
  2. List<String> list, String delimiter, String prefix, String suffix) {
  3. return list.stream().collect(Collectors.joining(delimiter, prefix, suffix));
  4. }
  5. public static void main(String[] args) throws Exception {
  6. List<String> list = Arrays.asList("a", "b", "c", "d", "e", "f", "g");
  7. System.out.println("使用 Collectors.joining:");
  8. String format = formatList(list, ", ", "{ ", " }");
  9. System.out.println(format);
  10. }

运行结果:
使用 Collectors.joining

查看 StringJoiner 的源码,我们可以知道 StringJoiner 的底层实现就是 StringBuilder —— Java8 将 使用分隔符连接多个字符串 这一功能进行封装,提供这么多易用且简介的 API,确实在很大程度上方便了编码

发表评论

表情:
评论列表 (有 0 条评论,389人围观)

还没有评论,来说两句吧...

相关阅读

    相关 Java 8改进类型推断

    一 点睛 Java8改进了泛型方法的类型推断能力,类型推断主要有如下两个方面。 可通过调用方法的上下文来推断类型参数的目标类型。 可在方法调用链中,将推断得到的类型参数传