3.字符串与容器互转

一灰灰blogJava编程技巧JDK编程技巧约 1028 字大约 3 分钟

实战3:字符串与Collection的互转

将字符串转换为List,这种业务场景可以说非常非常常见了,实现方式也比较简单

public List<String> str2list(String str, String split) {
    String[] cells = str.split(split);
    return Arrays.asList(cells);
}

那么除了上面这种实现方式之外,还有其他的么?

1. 字符串转列表

上面的实现姿势相当于字符串先转数组,然后在通过数组转列表,所以可以沿用前一篇字数组转list的几种方式

1.1. jdk支持方式

借助Collections.addAll来实现

public List<String> str2list2(String str, String split) {
    List<String> list = new ArrayList<>();
    Collections.addAll(list, str.split(split));
    return list;
}

上面这种方式适用于输出String的列表,如果我希望转成int列表呢?可以采用下面的方式

public List<Integer> str2intList(String str, String split) {
    return Stream.of(str.split(split))
            .map(String::trim)
            .filter(s -> !s.isEmpty())
            .map(Integer::valueOf).collect(Collectors.toList());
}

直接将数组转换为流,然后基于jdk8的特性,来实现转换为int列表

1.2. guava方式

引入依赖

<!-- https://mvnrepository.com/artifact/com.google.guava/guava -->
<dependency>
    <groupId>com.google.guava</groupId>
    <artifactId>guava</artifactId>
    <version>30.1-jre</version>
</dependency>

除了使用jdk原生的方式之外,借助guava也是非常常见的的case了,主要通过Splitter来实现,写法看起来非常秀

public List<String> str2list2(String str, String split) {
    return Splitter.on(split).splitToList(str);
}

简单直接的一行代码搞定,如果我们希望是对输出的列表类型进行指定,也可以如下操作

public List<Integer> str2intListV2(String str, String split) {
    return Splitter.on(split).splitToStream(str)
            .map(String::trim).filter(s -> !s.isEmpty())
            .map(Integer::valueOf).collect(Collectors.toList());
}

1.3. apache-commons

引入依赖

 <dependency>
    <groupId>org.apache.commons</groupId>
    <artifactId>commons-collections4</artifactId>
    <version>4.4</version>
</dependency>

上面流的方式就很赞了,但是注意它是有jdk版本限制的,虽说现在基本上都是1.8以上的环境进行开发,但也不排除有上古的代码,比如我现在手上的项目,spring还是3...

如果我们不能使用流的方式,那么有什么简单的方式来实现字符串转换为指定类型的列表么?

public List<Integer> str2intListV3(String str, String split) {
    List<Integer> result = new ArrayList<>();
    CollectionUtils.collect(Arrays.asList(str.split(split)), new Transformer<String, Integer>() {
        @Override
        public Integer transform(String s) {
            return Integer.valueOf(s);
        }
    }, result);
    return result;
}

上面这个实现也没有多优雅,不过这里有个编程小技巧可以学习,new Transformer(){}的传参方式,这种实现方式有点像回调的写法,虽然他们有本质的区别,此外就是jdk8之后的函数方法,就充分的体现这种设计思路,比如上面的换成jdk8的写法,直接简化为

public List<Integer> str2intListV3(String str, String split) {
    List<Integer> result = new ArrayList<>();
    CollectionUtils.collect(Arrays.asList(str.split(split)), Integer::valueOf, result);
    return result;
}

2. 列表转字符串

2.1. StringBuilder

最容易想到的,直接使用StringBuilder来实现拼接

public String list2str(List<String> list, String split) {
    StringBuilder builder = new StringBuilder();
    for (String str: list) {
        builder.append(str).append(split);
    }
    return builder.substring(0, builder.length() - 1);
}

注意两点:

  • 使用StringBuilder而不是StringBuffer (why?)
  • 注意最后一个拼接符号不要

2.2. String.join

一个更简单的实现方式如下

public String list2str2(List<String> list, String split) {
    return String.join(split, list);
}

当然上面这个的缺点就是列表必须是字符串列表,如果换成int列表,则不行

2.3. gauva

guava也提供了列表转String的方式,同样很简单,而且还没有列表类型的限制

public <T> String list2str3(List<T> list, String split) {
    return Joiner.on(split).join(list);
}

3. 小结

本文的考题也非常常见,列表与字符串的互转,这里介绍了多种实现方式,有jdk原生的case(如果没有什么限制,推荐使用它, String.split除外,原因后面再说),如果有更高级的定制场景,如非String类型类表,则可以考虑guava的Splitter/Joinner来实现

在上面的实现中,也提供了几种有意思的编程方式

  • Stream: 流,jdk8之后非常常见了
  • 函数方法,回调写法case
Loading...