Java:将List <String>转换为字符串


594

JavaScript有 Array.join()

js>["Bill","Bob","Steve"].join(" and ")
Bill and Bob and Steve

Java有这样的东西吗?我知道我可以使用StringBuilder自己整理一些东西:

static public String join(List<String> list, String conjunction)
{
   StringBuilder sb = new StringBuilder();
   boolean first = true;
   for (String item : list)
   {
      if (first)
         first = false;
      else
         sb.append(conjunction);
      sb.append(item);
   }
   return sb.toString();
}

...但是如果类似的东西已经成为JDK的一部分,那么这样做是没有意义的。


1
也看到了这个问题列表
Casebash

18
并非严格相关,但android在其TextUtils类中具有内置的join函数:developer.android.com/reference/android/text/…,java.lang.Iterable
Xavi

16
Java 8有一个String.join()方法。如果您使用的是Java 8(或更高版本),请看一下此答案stackoverflow.com/a/22577565/1115554
micha 2014年

Answers:


761

使用Java 8,您无需任何第三方库就可以做到这一点。

如果要加入字符串集合,可以使用新的String.join()方法:

List<String> list = Arrays.asList("foo", "bar", "baz");
String joined = String.join(" and ", list); // "foo and bar and baz"

如果您的Collection的类型不是String,则可以将Stream API与加入的Collector一起使用

List<Person> list = Arrays.asList(
  new Person("John", "Smith"),
  new Person("Anna", "Martinez"),
  new Person("Paul", "Watson ")
);

String joinedFirstNames = list.stream()
  .map(Person::getFirstName)
  .collect(Collectors.joining(", ")); // "John, Anna, Paul"

StringJoiner班也可能是有用的。


7
不幸的是,String.join仅接受CharSequences,而不像人们希望的那样接受Object。甚至不确定它是否为nullsafe
Marc

@MarcassertThat(String.join(", ", Lists.newArrayList("1", null)), is("1, null"));
Sanghyun Lee

如果想要加入[a, b, c]包括括号在内的东西,StringJoiner尤其有用。
koppor

@Marc String.join也接受Iterable<CharSequence>;界面关系:Iterable -> Collection -> List
aff

306

所有对Apache Commons的引用都很好(这是大多数人使用的),但我认为相当于Guava的Joiner具有更好的API。

您可以使用以下简单的连接案例

Joiner.on(" and ").join(names)

而且还可以轻松处理null:

Joiner.on(" and ").skipNulls().join(names);

要么

Joiner.on(" and ").useForNull("[unknown]").join(names);

(就我而言,优先于commons-lang使用就足够了)能够处理Maps:

Map<String, Integer> ages = .....;
String foo = Joiner.on(", ").withKeyValueSeparator(" is ").join(ages);
// Outputs:
// Bill is 25, Joe is 30, Betty is 35

这对于调试等非常有用


10
谢谢你的插头!我们的Joiner还为您提供了直接附加到Appendable(例如StringBuilder或任何Writer)的选项,而无需创建Apache库似乎缺少的中间字符串。
凯文·布劳里恩

2
很好,但是您可以添加一个.useForLastSeparator()或类似的方法吗?这样,您仅在最后两个项目之间可以得到类似“和”的信息(其余为“,”)。
杰夫·埃文斯

2
是的,它是线程安全的。联接器中保存的唯一状态是separatorfinal)。在Guava中,我曾经使用过Apache Commons等效项的一切都比在Guava中好得多(阅读:更干净,更快,更安全,总体上更加健壮),而且边缘情况失败和线程安全问题更少,并且较小的内存占用。到目前为止,他们的“最佳方式”指导原则似乎是正确的。
Shadow Man

如果您需要朝相反的方向移动(即,将一个字符串拆分成多个部分),请查看Guava类Splitter-也是非常好的设计/实现。
Matt Passell 2014年

在Java 8中,有一个String.join()方法和一个StringJoiner类。
theTechnoKid 2014年

138

虽然不是开箱即用的,但是许多库都有类似的功能:

公地郎:

org.apache.commons.lang.StringUtils.join(list, conjunction);

弹簧:

org.springframework.util.StringUtils.collectionToDelimitedString(list, conjunction);


49

不,标准Java API中没有这样的便捷方法。

毫不奇怪,如果您不想自己编写,Apache Commons 在其StringUtils类中提供了此类功能。


11
一直让我感到困惑的是,String具有拆分功能,但没有联接。从某种意义上说,它确实可行,但是令人讨厌的是,String中至少没有静态方法。
Powerlord

3
我和你在一起Bemrose。至少他们在... ...中给了我们一个isEmpty()方法,而不是(静态)join()方法String::rollseyes::)
Bart

41

Java 8中的三种可能性:

List<String> list = Arrays.asList("Alice", "Bob", "Charlie")

String result = String.join(" and ", list);

result = list.stream().collect(Collectors.joining(" and "));

result = list.stream().reduce((t, u) -> t + " and " + u).orElse("");

27

使用Java 8收集器,可以使用以下代码完成此操作:

Arrays.asList("Bill", "Bob", "Steve").stream()
.collect(Collectors.joining(" and "));

另外,Java 8中最简单的解决方案:

String.join(" and ", "Bill", "Bob", "Steve");

要么

String.join(" and ", Arrays.asList("Bill", "Bob", "Steve"));

21

我写了这个(我将它用于bean和exploit toString,所以不要写Collection<String>):

public static String join(Collection<?> col, String delim) {
    StringBuilder sb = new StringBuilder();
    Iterator<?> iter = col.iterator();
    if (iter.hasNext())
        sb.append(iter.next().toString());
    while (iter.hasNext()) {
        sb.append(delim);
        sb.append(iter.next().toString());
    }
    return sb.toString();
}

CollectionJSP不支持,因此对于TLD,我写道:

public static String join(List<?> list, String delim) {
    int len = list.size();
    if (len == 0)
        return "";
    StringBuilder sb = new StringBuilder(list.get(0).toString());
    for (int i = 1; i < len; i++) {
        sb.append(delim);
        sb.append(list.get(i).toString());
    }
    return sb.toString();
}

.tld存档:

<?xml version="1.0" encoding="UTF-8"?>
<taglib version="2.1" xmlns="http://java.sun.com/xml/ns/javaee"
    <function>
        <name>join</name>
        <function-class>com.core.util.ReportUtil</function-class>
        <function-signature>java.lang.String join(java.util.List, java.lang.String)</function-signature>
    </function>
</taglib>

并在JSP文件中将其用作:

<%@taglib prefix="funnyFmt" uri="tag:com.core.util,2013:funnyFmt"%>
${funnyFmt:join(books, ", ")}

1
建议更改Collection<>Iterable<>
杰森S

@JasonS +1。好点,但请阅读stackoverflow.com/questions/1159797/…–
Givenkoa

19

如果您想在没有任何外部库的情况下使用JDK,那么您拥有的代码就是正确的方法。在JDK中没有简单的“单一代码”。

如果可以使用外部库,建议您查看Apache Commons库中的org.apache.commons.lang.StringUtils类。

用法示例:

List<String> list = Arrays.asList("Bill", "Bob", "Steve");
String joinedResult = StringUtils.join(list, " and ");

17

一个正统的方法是通过定义一个新函数:

public static String join(String joinStr, String... strings) {
    if (strings == null || strings.length == 0) {
        return "";
    } else if (strings.length == 1) {
        return strings[0];
    } else {
        StringBuilder sb = new StringBuilder(strings.length * 1 + strings[0].length());
        sb.append(strings[0]);
        for (int i = 1; i < strings.length; i++) {
            sb.append(joinStr).append(strings[i]);
        }
        return sb.toString();
    }
}

样品:

String[] array = new String[] { "7, 7, 7", "Bill", "Bob", "Steve",
        "[Bill]", "1,2,3", "Apple ][","~,~" };

String joined;
joined = join(" and ","7, 7, 7", "Bill", "Bob", "Steve", "[Bill]", "1,2,3", "Apple ][","~,~");
joined = join(" and ", array); // same result

System.out.println(joined);

输出:

7、7、7以及Bill和Bob和Steve以及[Bill]和1,2,3以及Apple] [和〜,〜


5
为方法参数提供与方法本身相同的名称可能不是最好的主意。separator会更具启发性。
ccpizza 2015年

10

Java 8解决方案 java.util.StringJoiner

Java 8有一个StringJoiner类。但是您仍然需要编写一些样板文件,因为它是Java。

StringJoiner sj = new StringJoiner(" and ", "" , "");
String[] names = {"Bill", "Bob", "Steve"};
for (String name : names) {
   sj.add(name);
}
System.out.println(sj);

如果使用更方便的String.join()方法,则无需编写一些样板文件。
安迪·托马斯


7

你可以这样做:

String aToString = java.util.Arrays.toString(anArray);
// Do not need to do this if you are OK with '[' and ']'
aToString = aToString.substring(1, aToString.length() - 1);

或单线(仅当您不希望使用[[和]]时)

String aToString = java.util.Arrays.toString(anArray).substring(1).replaceAll("\\]$", "");

希望这可以帮助。


1
这不会附加选择的合取。
hexium

糟糕!对不起,我没发现。
NawaMan

6

在一个职责范围内使用纯JDK的有趣方法:

String[] array = new String[] { "Bill", "Bob", "Steve","[Bill]","1,2,3","Apple ][" };
String join = " and ";

String joined = Arrays.toString(array).replaceAll(", ", join)
        .replaceAll("(^\\[)|(\\]$)", "");

System.out.println(joined);

输出:

Bill and Bob and Steve and [Bill] and 1,2,3 and Apple] [


不太完美也不太有趣的方式!

String[] array = new String[] { "7, 7, 7","Bill", "Bob", "Steve", "[Bill]",
        "1,2,3", "Apple ][" };
String join = " and ";

for (int i = 0; i < array.length; i++) array[i] = array[i].replaceAll(", ", "~,~");
String joined = Arrays.toString(array).replaceAll(", ", join)
        .replaceAll("(^\\[)|(\\]$)", "").replaceAll("~,~", ", ");

System.out.println(joined);

输出:

7、7、7以及Bill和Bob和Steve以及[Bill]和1,2,3以及Apple] [


尝试使用“ [Bill]”,“ 1,2,3”和“ Apple] [”。有创造力,但有一些情况是错误的。
詹森·S

1
现在尝试使用“〜,〜”。您不能完全采用这种架构来编写程序以防弹。
詹森·S

是的,我知道...我已经删除了选票。但是,对于在此处发布答案,您应该多加考虑,尤其是对于已经存在数年的此类问题。答案不仅对原始海报立即有用,而且还会显示在Google搜索中。(实际上,对于已经存在数年的问题,新的答案可能对原始发布者没有任何价值。)存在缺陷或错误的解决方案可能对以后在上下文之外发现它的人们有害。
詹森·S

1
我在这里从我的其他人那里学到了很多东西,虽然不是完美的解决方案,但确实具有丰富的知识和侧面的思考。为他人放松,每个人都必须了解其代码的每一行都在做什么。请记住,编程就像一门艺术,甚至每一行代码也都意味着程序员的个性。是的,我不会在生产中使用此代码!
Daniel DeLeón13年


4

如果您使用的是Eclipse Collections(以前称为GS Collections),则可以使用该makeString()方法。

List<String> list = Arrays.asList("Bill", "Bob", "Steve");

String string = ListAdapter.adapt(list).makeString(" and ");

Assert.assertEquals("Bill and Bob and Steve", string);

如果您可以将其转换List为Eclipse Collections类型,则可以摆脱适配器。

MutableList<String> list = Lists.mutable.with("Bill", "Bob", "Steve");
String string = list.makeString(" and ");

如果只需要逗号分隔的字符串,则可以使用makeString()不带参数的版本。

Assert.assertEquals(
    "Bill, Bob, Steve", 
    Lists.mutable.with("Bill", "Bob", "Steve").makeString());

注意:我是Eclipse Collections的提交者。


3

使用java 1.8流时,

import java.util.Arrays;
import java.util.List;
import java.util.stream.Collectors;
List<String> list = Arrays.asList("Bill","Bob","Steve").
String str = list.stream().collect(Collectors.joining(" and "));

3

编辑

我还注意到了 toString()潜在的实现问题,以及有关包含分隔符的元素的问题,但我认为自己很偏执。

由于在这方面有两点评论,因此我将答案更改为:

static String join( List<String> list , String replacement  ) {
    StringBuilder b = new StringBuilder();
    for( String item: list ) { 
        b.append( replacement ).append( item );
    }
    return b.toString().substring( replacement.length() );
}

看起来与原始问题非常相似。

因此,如果您不想将整个jar添加到您的项目中,则可以使用它。

我认为您的原始代码没有错。实际上,每个人都建议的替代方案看起来几乎相同(尽管它进行了许多其他验证)

这就是 Apache 2.0许可证。

public static String join(Iterator iterator, String separator) {
    // handle null, zero and one elements before building a buffer
    if (iterator == null) {
        return null;
    }
    if (!iterator.hasNext()) {
        return EMPTY;
    }
    Object first = iterator.next();
    if (!iterator.hasNext()) {
        return ObjectUtils.toString(first);
    }

    // two or more elements
    StringBuffer buf = new StringBuffer(256); // Java default is 16, probably too small
    if (first != null) {
        buf.append(first);
    }

    while (iterator.hasNext()) {
        if (separator != null) {
            buf.append(separator);
        }
        Object obj = iterator.next();
        if (obj != null) {
            buf.append(obj);
        }
    }
    return buf.toString();
}

现在我们知道了,谢谢开源


现在可以使用,但是您不能确定List.toString()将来的行为。除了打印有关对象的一些信息性字符串外,我永远不会相信toString()操作。在您的辩护中,您并不是唯一提出此解决方案的人。
汉斯·多根

如果列表中元素的内容包含子字符串", "怎么办?
巴特·基尔斯

@汉斯和@巴特:你是对的。我以为没有人会注意到:P我正在改变我的答案。
OscarRyz

这取决于用例。出于调试目的,toString()只是IMO ...,只要您不依赖于特定的格式即可。即使这样,如果您为代码编写测试,将来任何更改都将被捕获。
akostadinov 2012年

2

Google的Guava API也具有.join(),尽管(在其他答复中应该很明显),但Apache Commons几乎是此处的标准。


1

Java 8确实带来了

Collectors.joining(CharSequence delimiter, CharSequence prefix, CharSequence suffix)

方法,即使用null prefix + suffix null值。

可以按以下方式使用它:

String s = stringList.stream().collect(Collectors.joining(" and ", "prefix_", "_suffix"))

Collectors.joining(CharSequence delimiter)方法仅在joining(delimiter, "", "")内部调用。


0

您可以从Spring Framework的StringUtils中使用它。我知道已经提到过它了,但是实际上您可以使用此代码即可立即运行,而无需Spring。

// from https://github.com/spring-projects/spring-framework/blob/master/spring-core/src/main/java/org/springframework/util/StringUtils.java

/*
 * Copyright 2002-2017 the original author or authors.
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *      http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */
public class StringUtils {
    public static String collectionToDelimitedString(Collection<?> coll, String delim, String prefix, String suffix) {
        if(coll == null || coll.isEmpty()) {
            return "";
        }
        StringBuilder sb = new StringBuilder();
        Iterator<?> it = coll.iterator();
        while (it.hasNext()) {
            sb.append(prefix).append(it.next()).append(suffix);
            if (it.hasNext()) {
                sb.append(delim);
            }
        }
        return sb.toString();
    }
}

-3

尝试这个:

java.util.Arrays.toString(anArray).replaceAll(", ", ",")
                .replaceFirst("^\\[","").replaceFirst("\\]$","");

1
可能是因为该问题要求使用列表而不是数组? 耸耸肩
Nick Coelius

4
因为如果您的字符串中有一个“,”将不起作用。
Cyrille Pontvieux
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.