Java: преобразовать список <String> в строку

596

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: developer.android.com/reference/android/text/… , java.lang.Iterable)
Хави
16
У Java 8 есть String.join()метод. Посмотрите на этот ответ, если вы используете Java 8 (или новее) stackoverflow.com/a/22577565/1115554
micha

Ответы:

764

С 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 принимает только CharSequence, а не так, как можно было бы надеяться на Object. Даже не уверен, что это нуль-сейф
Марк
@MarcassertThat(String.join(", ", Lists.newArrayList("1", null)), is("1, null"));
Санхьюн Ли
StringJoiner особенно полезен, если кто-то хочет присоединиться к чему-то вроде [a, b, c]включения скобок.
Коппор
@Marc String.join также принимает Iterable<CharSequence>; Отношения интерфейса:Iterable -> Collection -> List
aff
306

Все ссылки на Apache Commons в порядке (и это то, что большинство людей используют), но я думаю, что эквивалент Guava , Joiner , имеет гораздо более приятный API.

Вы можете сделать простой случай соединения с

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

но также легко справиться с нулями:

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

или

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

и (достаточно полезно, насколько я понимаю, чтобы использовать его вместо commons-lang), возможность работать с Картами:

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
Да, это потокобезопасно. Единственное состояние, сохраненное в столяре - это separator(что есть final). Все, что я видел в Гуаве, где я использовал эквивалент Apache Commons, было намного лучше (читай: чище, быстрее, безопаснее и просто более надежно в целом) в Гуаве с меньшим количеством сбоев в крайних случаях и проблемами безопасности потоков и меньший объем памяти. Похоже, что их руководящий принцип «наилучшим образом» пока остается в силе.
Shadow Man
Если вам нужно пойти в противоположном направлении (то есть разделить строку на части), посмотрите класс Splitter для Guava - также очень хорошо разработанный / реализованный.
Мэтт Пасселл
В Java 8 есть String.join()метод и StringJoinerкласс.
theTechnoKid
138

Не из коробки, но во многих библиотеках есть похожие:

Commons Lang:

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

Весна:

org.springframework.util.StringUtils.collectionToDelimitedString(list, conjunction);
Арне Бурмейстер
источник
125

На Android вы можете использовать класс TextUtils .

TextUtils.join(" and ", names);
Рафаль Энден
источник
11
Я искал это. String.joinтребует минимального уровня API 26 на Android.
Okarakose
49

Нет, в стандартном Java API такого удобного метода нет.

Неудивительно, что Apache Commons предоставляет такую ​​возможность в своем классе StringUtils на тот случай, если вы не хотите писать это самостоятельно.

Барт Киерс
источник
11
Меня всегда беспокоило, что у String есть разделение, но нет объединения. В каком-то смысле это имеет смысл, но просто раздражает, что в String нет по крайней мере статического метода для этого.
Powerlord
3
Я с тобой Бемроуз. По крайней мере, они дали нам isEmpty()метод вместо (статического) join()метода в String...: rollseyes: :)
Барт Киерс
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"));
AIFA
источник
21

Я написал это (я использую его для bean-компонентов и эксплойтов 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();
}

но Collectionне поддерживается JSP, поэтому для 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, ", ")}
gavenkoa
источник
1
предложить изменить Collection<>на Iterable<>?
Джейсон С
@JasonS +1. Хороший вопрос, но читайте stackoverflow.com/questions/1159797/…
gavenkoa
19

Код, который вы имеете, является правильным способом сделать это, если вы хотите использовать JDK без каких-либо внешних библиотек. Нет простого «однострочного», который вы могли бы использовать в JDK.

Если вы можете использовать внешние библиотеки , я рекомендую вам заглянуть в класс org.apache.commons.lang.StringUtils в библиотеке Apache Commons.

Пример использования:

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 и Билл и Боб и Стив и [Билл] и 1,2,3 и Apple] [и ~, ~

Даниэль Де Леон
источник
5
давать параметру метода то же имя, что и сам метод, вероятно, не лучшая идея. separatorбыло бы намного более наводящим на размышления.
ccpizza
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);
klingt.net
источник
Вам не нужно писать небольшой шаблон, если вы используете более удобный метод String.join () .
Энди Томас
9

Вы можете использовать библиотеку apache commons, которая имеет класс StringUtils и метод join.

Проверьте эту ссылку: https://commons.apache.org/proper/commons-lang/javadocs/api.2.0/org/apache/commons/lang/StringUtils.html

Обратите внимание, что приведенная выше ссылка может со временем устареть, и в этом случае вы можете просто выполнить поиск в сети «apache commons StringUtils», что должно позволить вам найти последнюю ссылку.

(ссылка из этого потока) Java-эквиваленты C # String.Format () и String.Join ()

DCP
источник
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("\\]$", "");

Надеюсь это поможет.

NawaMan
источник
1
Это не добавит сочетание выбора.
гексий
ОЙ !! Извините, я этого не заметил.
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);

Вывод:

Билл и Боб и Стив и [Билл] и 1,2,3 и 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 и Билл и Боб и Стив и [Билл] и 1,2,3 и Apple] [

Даниэль Де Леон
источник
Попробуйте это с помощью «[Bill]», «1,2,3» и «Apple] [». Креатив, но есть случаи, когда это неправильно.
Джейсон С
1
Теперь попробуйте это с "~, ~". Вы не можете сделать программу с такой архитектурой полностью пуленепробиваемой.
Джейсон С
Да, я знаю ... Я убрал понижающий голос. Но вы должны подумать о размещении ответов здесь, особенно для таких вопросов, которым несколько лет. Мало того, что ответы сразу полезны для оригинального постера, но они также отображаются в поиске Google. (На самом деле, по вопросам, которые несколько лет, новые ответы не могут иметь не ценность оригинального плаката.) Решения , которые имеют недостатки или ошибки могут быть вредны для людей , которые находят это позже из контекста.
Джейсон С
1
Я многому научился здесь от других постов, подобных моей, с не идеальным решением, но действительно богатым знанием и побочным мышлением. Расслабьтесь для других, каждый должен понимать, что делает каждая строка их кода. Помните, что программирование похоже на искусство, и даже каждая строка кода означает что-то о личности программиста. И да, я не буду использовать этот код в производстве!
Даниэль Де Леон
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, то вы можете избавиться от адаптера.

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() );
}

Который выглядит довольно похоже на оригинальный вопрос.

Так что, если вы не хотите добавлять банку в свой проект, вы можете использовать это.

Я думаю, что нет ничего плохого в вашем исходном коде. На самом деле, альтернатива, которую предлагают все, выглядит почти одинаково (хотя она и делает несколько дополнительных проверок)

Вот оно, вместе с лицензией 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();
}

Теперь мы знаем, спасибо с открытым исходным кодом

OscarRyz
источник
Это будет работать сейчас, но вы не можете быть уверены, как List.toString () будет вести себя в будущем. Я бы никогда не стал доверять операции toString (), кроме печати некоторой информационной строки о рассматриваемом объекте. В свою защиту вы не единственный, кто предлагает это решение.
Ганс Догген
Что, если содержимое элемента в списке содержит подстроку ", "?
Барт Киерс
@ Ганс и @ Барт: Ты прав. Я думал, что никто не заметит: P Я меняю свой ответ.
ОскарРиз
Это зависит от варианта использования. Для отладки toString () - это просто IMO ... при условии, что вы не полагаетесь на конкретное форматирование. И даже в этом случае, если вы напишете тесты для своего кода, любые будущие изменения будут обнаружены.
Акостадинов
2

Google Guava API также имеет .join (), хотя (как должно быть очевидно из других ответов), Apache Commons здесь в значительной степени стандарт.

Дин Дж
источник
1

Java 8 действительно приносит

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

метод, который является нулевым, используя prefix + suffixнулевые значения.

Его можно использовать следующим образом:

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

Collectors.joining(CharSequence delimiter)Метод просто вызывает joining(delimiter, "", "")внутренне.

THG
источник
0

Вы можете использовать это из StringUtils Spring Framework. Я знаю, что это уже упоминалось, но вы можете просто взять этот код, и он работает немедленно, без необходимости использования 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();
    }
}
EpicPandaForce
источник
-3

Попробуй это:

java.util.Arrays.toString(anArray).replaceAll(", ", ",")
                .replaceFirst("^\\[","").replaceFirst("\\]$","");
хорхе
источник
1
Возможно, потому что вопрос требует использования List вместо Array? пожимает плечами
Ник Coelius
4
Потому что если в ваших строках есть ",", это не сработает.
Сириль Понтвье