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