是否有任何简单的 LINQ 表达式可以将我的整个 List<string>
集合项连接到带有分隔符的单个 string
?
如果集合是自定义对象而不是 string
怎么办?想象一下我需要在 object.Name
上连接。
string result = String.Join(delimiter, list);
足够了。
警告 - 严重的性能问题
尽管此答案确实产生了预期的结果,但与此处的其他答案相比,它的性能较差。决定使用它时要非常小心
通过使用 LINQ,这应该可以工作;
string delimiter = ",";
List<string> items = new List<string>() { "foo", "boo", "john", "doe" };
Console.WriteLine(items.Aggregate((i, j) => i + delimiter + j));
班级说明:
public class Foo
{
public string Boo { get; set; }
}
用法:
class Program
{
static void Main(string[] args)
{
string delimiter = ",";
List<Foo> items = new List<Foo>() { new Foo { Boo = "ABC" }, new Foo { Boo = "DEF" },
new Foo { Boo = "GHI" }, new Foo { Boo = "JKL" } };
Console.WriteLine(items.Aggregate((i, j) => new Foo{Boo = (i.Boo + delimiter + j.Boo)}).Boo);
Console.ReadKey();
}
}
这是我最好的:)
items.Select(i => i.Boo).Aggregate((i, j) => i + delimiter + j)
注意:此答案不使用 LINQ 生成连接字符串。使用 LINQ 将枚举转换为分隔字符串可能会导致严重的性能问题
现代 .NET(自 .NET 4 起)
这适用于实现 IEnumerable 的数组、列表或任何类型:
string.Join(delimiter, enumerable);
这是用于可枚举的自定义对象:
string.Join(delimiter, enumerable.Select(i => i.Boo));
旧 .NET(.NET 4 之前)
这是一个字符串数组:
string.Join(delimiter, array);
这是一个列表<字符串>:
string.Join(delimiter, list.ToArray());
这是一个自定义对象列表:
string.Join(delimiter, list.Select(i => i.Boo).ToArray());
using System.Linq;
public class Person
{
string FirstName { get; set; }
string LastName { get; set; }
}
List<Person> persons = new List<Person>();
string listOfPersons = string.Join(",", persons.Select(p => p.FirstName));
好问题。我一直在使用
List<string> myStrings = new List<string>{ "ours", "mine", "yours"};
string joinedString = string.Join(", ", myStrings.ToArray());
它不是 LINQ,但它可以工作。
您可以简单地使用:
List<string> items = new List<string>() { "foo", "boo", "john", "doe" };
Console.WriteLine(string.Join(",", items));
快乐编码!
我认为如果您在扩展方法中定义逻辑,代码将更具可读性:
public static class EnumerableExtensions {
public static string Join<T>(this IEnumerable<T> self, string separator) {
return String.Join(separator, self.Select(e => e.ToString()).ToArray());
}
}
public class Person {
public string FirstName { get; set; }
public string LastName { get; set; }
public override string ToString() {
return string.Format("{0} {1}", FirstName, LastName);
}
}
// ...
List<Person> people = new List<Person>();
// ...
string fullNames = people.Join(", ");
string lastNames = people.Select(p => p.LastName).Join(", ");
List<string> strings = new List<string>() { "ABC", "DEF", "GHI" };
string s = strings.Aggregate((a, b) => a + ',' + b);
我已经使用 LINQ 做到了这一点:
var oCSP = (from P in db.Products select new { P.ProductName });
string joinedString = string.Join(",", oCSP.Select(p => p.ProductName));
将 String.Join
放入扩展方法中。这是我使用的版本,它比 Jordaos 版本更简洁。
当列表为空时返回空字符串“”。聚合会引发异常。
可能比聚合更好的性能
与其他 LINQ 方法结合使用时比纯 String.Join() 更易于阅读
用法
var myStrings = new List<string>() { "a", "b", "c" };
var joinedStrings = myStrings.Join(","); // "a,b,c"
扩展方法类
public static class ExtensionMethods
{
public static string Join(this IEnumerable<string> texts, string separator)
{
return String.Join(separator, texts);
}
}
此答案旨在扩展和改进对基于 LINQ 的解决方案的一些提及。这本身并不是解决此问题的“好”方法的示例。只需在适合您的需要时使用 string.Join
as suggested。
语境
这个答案是由问题的第二部分(一种通用方法)和一些表达对 LINQ 的深厚亲和力的评论引起的。
当前接受的答案似乎不适用于空序列或单例序列。它还存在性能问题。
当 ToString 没有产生所需的结果时,当前最受好评的答案没有明确解决通用字符串转换要求。 (这可以通过添加对 Select 的调用来解决。)
另一个答案包括可能导致一些人相信性能问题是 LINQ 固有的注释。 (“使用 LINQ 将枚举转换为分隔字符串可能会导致严重的性能问题。”)
我注意到这条关于将查询发送到数据库的评论。
鉴于没有满足所有这些要求的答案,我提出了一个基于 LINQ 的实现,以线性时间运行,使用任意长度的枚举,并支持将元素的通用转换为字符串。
那么,LINQ 还是破产?好的。
static string Serialize<T>(IEnumerable<T> enumerable, char delim, Func<T, string> toString)
{
return enumerable.Aggregate(
new StringBuilder(),
(sb, t) => sb.Append(toString(t)).Append(delim),
sb =>
{
if (sb.Length > 0)
{
sb.Length--;
}
return sb.ToString();
});
}
这种实现比许多替代方案更复杂,主要是因为我们需要在我们自己的代码中管理定界符(分隔符)的边界条件。
它应该以线性时间运行,最多遍历元素两次。
一次用于首先生成要附加的所有字符串,然后在最后一次 ToString
调用期间生成最终结果时从零到一次。这是因为后者可能只返回恰好足够大以包含从一开始就包含所有附加字符串的缓冲区,或者它必须重新生成完整的东西(不太可能),或者介于两者之间。有关详细信息,请参阅例如关于 SO 的 What is the Complexity of the StringBuilder.ToString()。
最后的话
如果适合您的需要,只需使用 string.Join
as suggested,当您需要先按摩序列时添加 Select
。
该答案的主要目的是说明可以使用 LINQ 来检查性能。结果(可能)过于冗长而无法推荐,但它确实存在。
您可以使用 Aggregate
将字符串连接成单个字符分隔的字符串,但如果集合为空,则会抛出 Invalid Operation Exception。
您可以将 Aggregate
函数与 seed 字符串一起使用。
var seed = string.Empty;
var seperator = ",";
var cars = new List<string>() { "Ford", "McLaren Senna", "Aston Martin Vanquish"};
var carAggregate = cars.Aggregate(seed,
(partialPhrase, word) => $"{partialPhrase}{seperator}{word}").TrimStart(',');
您可以使用 string.Join
不在乎是否将空集合传递给它。
var seperator = ",";
var cars = new List<string>() { "Ford", "McLaren Senna", "Aston Martin Vanquish"};
var carJoin = string.Join(seperator, cars);
<String>
以外的其他内容,并将调用 ToString() 方法。List<string>
绝不是进入数据库的查询。这是一个完全不同的问题,但您始终可以调用.ToList()
进行查询并稍后合并。