从对象列表中创建逗号分隔的列表

5
我有一个Person对象列表,这个类有很多属性,我需要将它们全部以逗号分隔的形式列出来,这样我之后可以用于Csv文件。我通过foreach循环和手动添加每个属性,并用逗号分隔来实现这个目标。
const string commaSeparator = ",";
foreach (var item in individualsInformation)
{
    csv.AppendLine(item.ReferenceNumber + commaSeparator + item.FirstName + commaSeparator +
                   item.Surname + commaSeparator + item.MiddleName + commaSeparator +
                   item.Address1 + commaSeparator + item.Address2 + commaSeparator + 
                   item.Address3 + commaSeparator + item.Address4 + commaSeparator + 
                   item.City + commaSeparator + item.PostalCode + commaSeparator +
                   item.Country + commaSeparator + item.DateOfBirth.ToString() + commaSeparator +
                   item.ID + commaSeparator + item.Gender + commaSeparator +
                   item.Component + commaSeparator + item.NationalID + commaSeparator + 
                   item.SubSystemID + commaSeparator + item.System);
}

然后我意识到有更高效的方法,可以使用string.Join

当然,这不起作用:

string joined = string.Join(",", listOfPersons);

如果我按照这样选择属性:

string joined = string.Join(",", listOfPersons(x => x.Id);

当然,我只得到逗号分隔的属性列表。

有没有更有效的方法来获取每个用逗号分隔的属性?


1
停止试图重新发明轮子。在这里检查答案:https://dev59.com/AF8e5IYBdhLWcg3wfajv#25684175 - user1641172
1
你目前的方法并不全是坏的,也不是拥有对一切完全控制的唯一方式。 - H H
4个回答

18

如果可能的话,我会避免使用反射。

你可以轻松地通过编译时安全性和可读性来实现:

IEnumerable<string> personTexts = listOfPersons
    .Select(p => String.Join(",", p.ReferenceNumber, p.FirstName, p.Surname, ...));
string joined = String.Join(Environment.NewLine, personTexts);
你完全可以控制哪个属性应该被使用以及它们的顺序。


7

反射有时是你的朋友:

var props = item.GetType().GetProperties(BindingFlags.Public | BindingFlags.Instance); // todo: cache & filter not-needed props)

var itemStr = string.Join(", ", 
                     props.Select(p => p.GetValue(item, null)?.ToString())
                          .ToArray());

反射确实帮了大忙。谢谢! :) @nopeflow - nemo_87

4

为您的Person类重写ToString方法

public class Person
{
    //properties...

    public override string ToString()
    {
        return string.Join(",",
            this.ReferenceNumber,
            this.FirstName,
            this.Surname,
            this.MiddleName,
            this.Address1,
            this.Address2,
            this.Address3,
            this.Address4,
            this.City,
            this.PostalCode,
            this.Country,
            this.DateOfBirth.ToString(),
            this.ID,
            this.Gender,
            this.Component,
            this.NationalID,
            this.SubSystemID,
            this.System);
    }
}

因此,您可以使用Person.ToString()来生成csv。与反射方法不同的是,您可以轻松

  • 维护顺序
  • 处理每种类型的不同之处(BirthDate.ToString("d")Price.ToString("F2")
  • 跳过不需要的值,例如System.Collections.Generic.List1[System.String]
  • 根据您的要求更改值(HasChildren ? "Yes" : "No"

0

使用反射 where T : class

var valueLines = reportData.Select(row => string.Join(",", header.Split(',').Select(a => row.GetType().GetProperty(a).GetValue(row, null))));

例如

private MemoryStream GenerateExcelStream(List<T> reportData)
    {
        var lines = new List<string>();
        var header = "";
        var attFilter = new NoDisplayInReportAttribute();
        PropertyDescriptorCollection properties = TypeDescriptor.GetProperties(typeof(T));
        foreach (PropertyDescriptor prop in properties)
            if (!prop.Attributes.Matches(attFilter))
                header += prop.Name + ",";
        header = header.Substring(0, header.Length - 1);
        lines.Add(header);
        var valueLines =reportData.Select(row => string.Join(",", header.Split(',').Select(a => row.GetType().GetProperty(a).GetValue(row, null))));
        lines.AddRange(valueLines);
        MemoryStream memoryStream = new MemoryStream();
        TextWriter tw = new StreamWriter(memoryStream);
        lines.ForEach(x => tw.WriteLine(x));
        tw.Flush();
        return memoryStream;
    }

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接