Java: แปลง List <String> เป็น String


595

JavaScript มี Array.join()

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

Java มีอะไรเช่นนี้หรือไม่? ฉันรู้ว่าฉันสามารถ cobble บางสิ่งบางอย่างตัวเองด้วย 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)
Xavi

16
Java 8 มีString.join()วิธีการ ดูที่คำตอบนี้หากคุณใช้ Java 8 (หรือใหม่กว่า) stackoverflow.com/a/22577565/1115554
micha

คำตอบ:


763

ด้วย Java 8 คุณสามารถทำได้โดยไม่ต้องมีห้องสมุดบุคคลที่สาม

หากคุณต้องการเข้าร่วม Collection of Strings คุณสามารถใช้เมธอด String.join ()ใหม่:

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

ถ้าคุณมีคอลเลกชันที่มีประเภทกว่า String อื่นที่คุณสามารถใช้ API สตรีมกับการเข้าร่วมสะสม :

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 และไม่เป็นไปตามที่หวังวัตถุ ไม่แน่ใจเหมือนกันว่ามันจะเป็นโมฆะหรือไม่
Marc

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

StringJoiner มีประโยชน์อย่างยิ่งหากต้องการเข้าร่วมกับสิ่งที่ต้องการ[a, b, c]รวมถึงการจัดฟัน
koppor

@Marc String.join ยังยอมรับIterable<CharSequence>; ความสัมพันธ์ของอินเตอร์เฟซ:Iterable -> Collection -> List
AFF

306

การอ้างอิงถึง Apache Commons นั้นใช้ได้ทั้งหมด (และนั่นคือสิ่งที่คนส่วนใหญ่ใช้) แต่ฉันคิดว่าGuava ที่เทียบเท่าJoinerมี API ที่ดีกว่ามาก

คุณสามารถทำกรณีเข้าร่วมง่ายๆด้วย

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

แต่ยังจัดการกับ nulls ได้อย่างง่ายดาย:

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

หรือ

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

และ (มีประโยชน์มากพอ ๆ กับที่ฉันใช้ในการตั้งค่าคอมมอนส์) ความสามารถในการจัดการกับแผนที่:

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

ซึ่งมีประโยชน์อย่างมากสำหรับการแก้ไขข้อบกพร่อง ฯลฯ


10
ขอบคุณสำหรับปลั๊ก! ช่างไม้ของเรายังให้คุณเลือกผนวกโดยตรงกับ Appendable (เช่น StringBuilder หรือ Writer) โดยไม่ต้องสร้าง Strings ระดับกลางซึ่งห้องสมุด Apache ดูเหมือนจะขาดไป
Kevin Bourrillion

2
นี่เป็นวิธีที่ดี แต่คุณสามารถเพิ่ม.useForLastSeparator()วิธีที่คล้ายกันได้ไหม ด้วยวิธีนี้คุณสามารถได้รับบางสิ่งเช่น "และ" ระหว่างสองรายการสุดท้ายเท่านั้น (กับ "," สำหรับส่วนที่เหลือ)
Jeff Evans

2
ใช่มันปลอดภัยสำหรับเธรด สถานะเดียวที่บันทึกในผู้เข้าร่วมคือseparator(ซึ่งก็คือfinal) ทุกสิ่งที่ฉันเห็นใน Guava ที่ฉันเคยใช้ Apache Commons เทียบเท่านั้นดีกว่ามาก (อ่าน: สะอาดเร็วขึ้นปลอดภัยขึ้นและแข็งแกร่งขึ้นโดยทั่วไป) ใน Guava ที่มีปัญหาเล็ก ๆ น้อย ๆ และปัญหาความปลอดภัยของเธรดและ หน่วยความจำขนาดเล็ก "แนวทางที่ดีที่สุดเท่าที่จะเป็นไปได้" ผู้แนะนำหลักดูเหมือนว่าจะเป็นจริง
Shadow Man

หากคุณต้องการที่จะไปในทิศทางตรงกันข้าม (เช่นการแยกสตริงออกเป็นชิ้น ๆ ) ตรวจสอบตัวแยกคลาส Guava - ซึ่งได้รับการออกแบบ / นำมาใช้เป็นอย่างดี
Matt Passell

ใน Java 8 มีString.join()วิธีและStringJoinerคลาส
theTechnoKid

138

ไม่ได้ออกนอกกรอบ แต่ห้องสมุดหลายแห่งมีลักษณะคล้ายกัน:

คอมมอนส์หรั่ง:

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ต้องการmin api level 26บน Android
okarakose

49

ไม่ไม่มีวิธีการอำนวยความสะดวกดังกล่าวใน Java API มาตรฐาน

ไม่น่าแปลกใจ Apache Commons ให้สิ่งนั้นในคลาส StringUtils ของพวกเขาในกรณีที่คุณไม่ต้องการเขียนด้วยตนเอง


11
มันบั๊กกับฉันเสมอว่า String มีการแบ่ง แต่ไม่ใช่การเข้าร่วม ด้วยวิธีการที่เหมาะสม แต่ก็เป็นเรื่องน่ารำคาญที่อย่างน้อยก็ไม่มีวิธีการคงที่ใน String
Powerlord

3
ฉันอยู่กับคุณ Bemrose อย่างน้อยพวกเขาให้เราisEmpty()วิธีการแทน (คงที่) join()วิธีการในString... : rollseyes: :)
บาร์ต Kiers

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

ฉันเขียนอันนี้ (ฉันใช้สำหรับถั่วและใช้ประโยชน์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, ", ")}

1
แนะนำให้เปลี่ยนCollection<>เป็นIterable<>?
Jason S

@JasonS +1 จุดดี แต่อ่านstackoverflow.com/questions/1159797/…
gavenkoa

19

รหัสที่คุณมีนั้นถูกต้องหากคุณต้องการใช้ JDK โดยไม่ต้องมีไลบรารีภายนอก ไม่มี "หนึ่งซับ" ง่าย ๆ ที่คุณสามารถใช้ใน JDK

หากคุณสามารถใช้ libs ภายนอกได้ฉันขอแนะนำให้คุณดูคลาส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 และ Bill and Bob และ Steve และ [Bill] และ 1,2,3 และ Apple] [และ ~, ~


5
การกำหนดพารามิเตอร์ method ชื่อเดียวกับ method นั้นอาจไม่ใช่ความคิดที่ดีที่สุด separatorจะมีคำแนะนำมากขึ้น
ccpizza

10

โซลูชัน Java 8 พร้อม java.util.StringJoiner

Java 8 มีStringJoinerคลาส แต่คุณยังคงต้องเขียนแผ่นความร้อนเพราะมันเป็นจาวา

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

คุณไม่จำเป็นต้องเขียนบิตของสำเร็จรูปถ้าคุณใช้วิธีการที่สะดวกมากขึ้นString.Join ()
Andy Thomas

9

คุณสามารถใช้ไลบรารี apache คอมมอนส์ซึ่งมีคลาส StringUtils และวิธีการเข้าร่วม

ตรวจสอบลิงค์นี้: 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 ()


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

OOPS !! ฉันขอโทษฉันไม่เห็น
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 และ Steve และ [Bill] และ 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 และ Bill and Bob และ Steve และ [Bill] และ 1,2,3 และ Apple] [


ลองด้วย "[Bill]", "1,2,3" และ "Apple] [" โฆษณา แต่มีหลายกรณีที่ไม่ถูกต้อง
Jason S

1
ตอนนี้ลองด้วย "~, ~" คุณไม่สามารถสร้างโปรแกรมด้วยสถาปัตยกรรมชนิดนี้กันกระสุนได้อย่างสมบูรณ์
Jason S

ใช่ฉันรู้ว่าฉันได้ลบข้อสงสัยออกแล้ว แต่คุณควรคิดให้รอบคอบมากขึ้นเกี่ยวกับการโพสต์คำตอบที่นี่โดยเฉพาะอย่างยิ่งสำหรับคำถามเช่นนี้ที่มีอายุหลายปี คำตอบไม่เพียง แต่เป็นประโยชน์ต่อผู้ลงประกาศดั้งเดิมเท่านั้น แต่ยังแสดงในการค้นหาของ Google (อันที่จริงสำหรับคำถามที่มีอายุหลายปีคำตอบใหม่อาจไม่มีประโยชน์กับโปสเตอร์ต้นฉบับ) วิธีแก้ปัญหาที่มีข้อบกพร่องหรือข้อผิดพลาดอาจเป็นอันตรายต่อผู้ที่พบในภายหลังนอกบริบท
Jason S

1
ฉันเรียนรู้ที่นี่มากมายจากคนอื่น ๆ โพสต์เหมือนของฉันด้วยโซลูชั่นที่ไม่สมบูรณ์แบบ แต่เต็มไปด้วยความรู้และการคิดข้างเคียง ผ่อนคลายเพื่อคนอื่น ๆ แต่ละคนจะต้องเข้าใจว่าแต่ละบรรทัดของโค้ดทำอะไร การจำการวางโปรแกรมเป็นเหมือนศิลปะและแม้แต่รหัสแต่ละบรรทัดก็มีความหมายบางอย่างเกี่ยวกับบุคลิกภาพของโปรแกรมเมอร์ และใช่ฉันจะไม่ใช้รหัสนี้ในการผลิต!
Daniel De León


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

ซึ่งมีลักษณะคล้ายกับคำถามเดิม

ดังนั้นหากคุณไม่รู้สึกอยากเพิ่มไหทั้งหมดในโครงการของคุณคุณสามารถใช้มันได้

ฉันคิดว่าไม่มีอะไรผิดปกติกับรหัสต้นฉบับของคุณ ที่จริงแล้วทางเลือกที่ทุกคนแนะนำดูเหมือนเกือบจะเหมือนกัน (แม้ว่าจะมีการตรวจสอบเพิ่มเติมจำนวนหนึ่ง)

นี่คือพร้อมกับ ใบอนุญาต 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 () นอกเหนือจากการพิมพ์สตริงที่ให้ข้อมูลเกี่ยวกับวัตถุที่เป็นปัญหา ในการป้องกันของคุณคุณไม่ใช่คนเดียวที่เสนอทางออกนี้
Hans Doggen

เกิดอะไรขึ้นถ้าเนื้อหาขององค์ประกอบในรายการมีซับสตริง", "หรือไม่
46432 Bart Kiers

@Hans & @Bart: คุณพูดถูก ฉันคิดว่าจะไม่มีใครสังเกตเห็น: P ฉันเปลี่ยนคำตอบของฉัน
OscarRyz

มันขึ้นอยู่กับกรณีการใช้งาน สำหรับวัตถุประสงค์ในการดีบั๊ก toString () เป็นเพียง IMO ... ที่คุณไม่ต้องพึ่งพาการจัดรูปแบบเฉพาะ และหากเป็นเช่นนั้นหากคุณเขียนการทดสอบสำหรับรหัสของคุณการเปลี่ยนแปลงใด ๆ ในอนาคตจะถูกตรวจจับ
akostadinov

2

API ของ Guava ของ Google ยังมี. join () แม้ว่า (ตามที่ควรจะเห็นได้ชัดเจนเมื่อมีคำตอบอื่น ๆ ) Apache Commons ก็เป็นมาตรฐานที่นี่


1

Java 8 นำมาซึ่ง

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

วิธีการนั่นคือ nullsafe โดยใช้prefix + suffixสำหรับค่า Null

มันสามารถใช้ในลักษณะดังต่อไปนี้:

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

Collectors.joining(CharSequence delimiter)วิธีการเพียงแค่เรียกร้องjoining(delimiter, "", "")ภายใน


0

คุณสามารถใช้สิ่งนี้ได้จาก StringUtils ของ Spring Framework ฉันรู้ว่ามันถูกกล่าวถึงแล้ว แต่คุณสามารถนำรหัสนี้มาใช้ได้จริงโดยไม่ต้องใช้สปริงสำหรับมัน

// 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
โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.