Java8:对字符串连接的改进

Java8:对字符串连接的改进
我们提出一个需求:有一个 List<String>,将其格式化为 元素1, 元素2, 元素3, ... 元素N 的字符串形式。

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

public static String formatList(List<String> list, String delimiter) {
    StringBuilder result = new StringBuilder();
    for (String str : list) {
        result.append(str).append(delimiter);
    }
    // 删除末尾多余的 delimiter
    result.delete(result.length() - delimiter.length(), result.length()); 
    
    return result.toString();
}

public static void main(String[] args) throws Exception {
    List<String> list = Arrays.asList("a", "b", "c", "d", "e", "f", "g");

    System.out.println("使用 StringBuilder:");
    String format = formatList(list, ", ");
    System.out.println(format);
}
复制代码

运行结果:

使用 StringBuilder

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

public static String formatList(List<String> list, String delimiter) {
    StringJoiner result = new StringJoiner(delimiter);
    for (String str : list) {
        result.add(str);
    }
    return result.toString();
}

public static void main(String[] args) throws Exception {
    List<String> list = Arrays.asList("a", "b", "c", "d", "e", "f", "g");

    System.out.println("使用 StringJoiner:");
    String format = formatList(list, ", ");
    System.out.println(format);
}

复制代码

运行结果:

使用 StringJoiner

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

String的 join 方法

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

public static String formatList(List<String> list, String delimiter) {
    return String.join(delimiter, list);
}
复制代码

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

String.join 的源码

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

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

StringJoiner 的构造方法

修改 formatList

public static String formatList(
        List<String> list, String delimiter, String prefix, String suffix) {

    StringJoiner result = new StringJoiner(delimiter, prefix, suffix);
    for (String str : list) {
        result.add(str);
    }
    return result.toString();
}

public static void main(String[] args) throws Exception {
    List<String> list = Arrays.asList("a", "b", "c", "d", "e", "f", "g");

    System.out.println("使用 StringJoiner,带前缀和后缀:");
    String format = formatList(list, ", ", "{ ", " }");
    System.out.println(format);
}
复制代码

运行结果:

StringJoiner,带前缀和后缀
很棒~ 但能不能更简洁呢? 事实上, 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:

public static String formatList(
        List<String> list, String delimiter, String prefix, String suffix) {

    return list.stream().collect(Collectors.joining(delimiter, prefix, suffix));
}

public static void main(String[] args) throws Exception {
    List<String> list = Arrays.asList("a", "b", "c", "d", "e", "f", "g");

    System.out.println("使用 Collectors.joining:");
    String format = formatList(list, ", ", "{ ", " }");
    System.out.println(format);
}
复制代码

运行结果:

使用 Collectors.joining

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

推荐

大厂笔试内容集合(内有详细解析) 持续更新中....

ProcessOn是一个在线作图工具的聚合平台~

文末

欢迎关注Coder编程公众号,主要分享数据结构与算法、Java相关知识体系、框架知识及原理、Spring全家桶、微服务项目实战、DevOps实践之路、每日一篇互联网大厂面试或笔试题以及PMP项目管理知识等。更多精彩内容正在路上~ 新建了一个qq群:315211365,欢迎大家进群交流一起学习。谢谢了!也可以介绍给身边有需要的朋友。

文章收录至 Github: github.com/CoderMerlin… Gitee: gitee.com/573059382/c… 欢迎关注并star~

微信公众号

猜你喜欢

转载自juejin.im/post/5db065515188256468478726