ChatGPT解决这个技术问题 Extra ChatGPT

使用 LINQ 连接 List<string> 中的所有字符串

是否有任何简单的 LINQ 表达式可以将我的整个 List<string> 集合项连接到带有分隔符的单个 string

如果集合是自定义对象而不是 string 怎么办?想象一下我需要在 object.Name 上连接。

为什么是 linq 而不是 string.Join() ?
string.Join 更好,但我认为 linq 让您的代码变得有趣,这可能就是原因!
String.Join 更好,因为它使用 StringBuilder 并避免了重复连接的固有 O(n^2) 性能。
使用 LINQ 的性能问题?

S
Sedat Kapanoglu
string result = String.Join(delimiter, list);

足够了。


我完全支持 LINQ 解决方案,但这比 LINQ 和 Aggregate() 方法更有效。
干净多了!对我来说很棒! string.Join(", ", accs.Select(x => x.AccountID).ToArray()),
@KonstantinSalavatov 在 OP 澄清它必须在 LINQ 中之前,我已经发布了我的答案。对于在 Google 上寻找“非必要 LINQ”解决方案时遇到此答案的任何人来说,它仍然是完全有效的。在这种情况下,关于这个答案“没有用”是不公平的。
这也可用于 List<String> 以外的其他内容,并将调用 ToString() 方法。
@alansiqueira27 好吧,List<string> 绝不是进入数据库的查询。这是一个完全不同的问题,但您始终可以调用 .ToList() 进行查询并稍后合并。
C
Caius Jard

警告 - 严重的性能问题

尽管此答案确实产生了预期的结果,但与此处的其他答案相比,它的性能较差。决定使用它时要非常小心

通过使用 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)

O(n^2) 时间再次来袭。
如果看不到 Aggregate 方法,则需要使用 System.Linq 添加;
问题是上述 LinQ 方法不适用于空列表或单元素列表。
为什么不直接使用 string.join?请接受 Sedat 的回答,这样当 Sedat 是更好的选择时,任何匆忙的人都不会选择此解决方案。
不要使用这个。该解决方案将立即降低应用程序性能,即使是在一组微不足道的字符串上。将 Sedat 的答案与 string.Join 一起使用!
C
Caius Jard

注意:此答案不使用 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());

String.Join 有一个采用 IEnumerable 的重载,因此您不需要 ToArray() 调用
请记住,IEnumerable 重载仅存在于 4.0 或更高版本中。如果您使用的是旧版本,您仍然需要 ToArray()。
啊!最后一个超载是我正在寻找的那个。我知道必须有一种方法来提取特定属性。 :)
d
dev.bv
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));

J
Jacob Proffitt

好问题。我一直在使用

List<string> myStrings = new List<string>{ "ours", "mine", "yours"};
string joinedString = string.Join(", ", myStrings.ToArray());

它不是 LINQ,但它可以工作。


为什么你必须调用 .ToArray() ?
因为在 2009 年糟糕的过去,string.Join 没有接受 IEnumerable 的扩展。
P
Peter Mortensen

您可以简单地使用:

List<string> items = new List<string>() { "foo", "boo", "john", "doe" };

Console.WriteLine(string.Join(",", items));

快乐编码!


J
Jordão

我认为如果您在扩展方法中定义逻辑,代码将更具可读性:

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(", ");

P
Peter McG
List<string> strings = new List<string>() { "ABC", "DEF", "GHI" };
string s = strings.Aggregate((a, b) => a + ',' + b);

P
Peter Mortensen

我已经使用 LINQ 做到了这一点:

var oCSP = (from P in db.Products select new { P.ProductName });

string joinedString = string.Join(",", oCSP.Select(p => p.ProductName));

W
Welcor

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

R
Robrecht Dewaele

此答案旨在扩展和改进对基于 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 来检查性能。结果(可能)过于冗长而无法推荐,但它确实存在。


R
Reza Jenabi

您可以使用 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);