在我使用数据库的过程中,我注意到我编写了查询字符串,在这个字符串中,我必须在list/array/collection的where子句中加入一些限制.应该是这样的:
select * from customer where customer.id in (34, 26, ..., 2);
您可以通过将此问题简化为具有字符串集合的问题并希望仅在一个字符串中创建此字符串的逗号分隔列表来简化此操作.
我到目前为止使用的方法是这样的:
String result = ""; boolean first = true; for(String string : collectionOfStrings) { if(first) { result+=string; first=false; } else { result+=","+string; } }
但这是你可以看到非常丑陋的.您无法看到第一眼看到的情况,特别是当构造的字符串(如每个SQL查询)变得复杂时.
你(更)优雅的方式是什么?
使用Google Guava API的join
方法:
Joiner.on(",").join(collectionOfStrings);
由于字符串是不可变的,因此如果您要更改代码中的String,则可能需要使用StringBuilder类.
StringBuilder类可以看作是一个可变的String对象,它在更改内容时分配更多的内存.
通过处理冗余的尾随逗号,可以更清楚有效地编写问题中的原始建议:
StringBuilder result = new StringBuilder(); for(String string : collectionOfStrings) { result.append(string); result.append(","); } return result.length() > 0 ? result.substring(0, result.length() - 1): "";
我刚看了今天这样做的代码.这是AviewAnew答案的变体.
collectionOfStrings = /* source string collection */; String csList = StringUtils.join(collectionOfStrings.toArray(), ",");
我们使用的StringUtils(< - commons.lang 2.x或commons.lang 3.x链接)来自Apache Commons.
我写这个循环的方式是:
StringBuilder buff = new StringBuilder(); String sep = ""; for (String str : strs) { buff.append(sep); buff.append(str); sep = ","; } return buff.toString();
不要担心sep的性能.作业非常快.无论如何,Hotspot倾向于剥离循环的第一次迭代(因为它经常需要处理诸如null和单/双态内联检查之类的奇怪现象).
如果您使用它(不止一次),请将其放在共享方法中.
stackoverflow还有另一个问题,涉及如何将一个id列表插入到SQL语句中.
从Java 8开始,您可以使用:
String String.join(CharSequence delimiter, CharSequence... elements)
String String.join(CharSequence delimiter, Iterable extends CharSequence> elements)
如果你想取非String
s并将它们加入a String
,你可以使用Collectors.joining(CharSequence delimiter)
,例如:
String joined = anyCollection.stream().map(Object::toString).collect(Collectors.joining(","));
我发现iterator成语优雅,因为它有更多元素的测试(ommited null/empty test为简洁起见):
public static String convert(Listlist) { String res = ""; for (Iterator iterator = list.iterator(); iterator.hasNext();) { res += iterator.next() + (iterator.hasNext() ? "," : ""); } return res; }
有很多手动解决方案,但我想重申和更新上面的Julie的答案.使用谷歌收藏品Joiner课程.
Joiner.on(", ").join(34, 26, ..., 2)
它处理var args,iterables和数组,并正确处理多个char的分隔符(与gimmel的答案不同).如果需要,它还将处理列表中的空值.
这是一个令人难以置信的通用版本,我是根据以前的建议组合而成的:
public staticString buildCommaSeparatedString(Collection values) { if (values==null || values.isEmpty()) return ""; StringBuilder result = new StringBuilder(); for (T val : values) { result.append(val); result.append(","); } return result.substring(0, result.length() - 1); }
String.join(", ", collectionOfStrings)
可以在Java8 api中找到.
替代(不需要添加谷歌番石榴依赖):
Joiner.on(",").join(collectionOfStrings);
你可以试试
List collections = Arrays.asList(34, 26, "...", 2); String asString = collection.toString(); // justValues = "34, 26, ..., 2" String justValues = asString.substring(1, asString.length()-1);