【问题标题】:How to Pivot IEnumerable<T>如何旋转 IEnumerable<T>
【发布时间】:2010-12-17 22:18:07
【问题描述】:

我需要“旋转”或“旋转”一组对象。该示例将清除我需要做的事情:

var people = new List<Person>(new[] {
                new Person{ Name="Ronnie",Age=25, HairColor="Brown",EyeColor="Blue"},
                new Person{ Name="Tina",Age=25, HairColor="Brown",EyeColor="Green"},
                new Person{ Name="Lukus",Age=4, HairColor="Blonde",EyeColor="Blue"}    
            });

DataTable rotatedData = people.Pivot("Name", "Property");                

/* Results In:
* ___________________________________________
* Property  |  Ronnie  |   Tina    |   Lukus
* ___________________________________________
* Age       |  25      |   25      |   4
* HairColor |  Brown   |   Brown   |   Blonde
* EyeColor  |  Blue    |   Green   |   Blue
* ____________________________________________
*/

这是我编写的代码:

/// <summary>Converts an IEnumerable into a pivoted DataTable object.</summary>
/// <param name="collection">The IEnumerable object to pivot and convert.</param>
/// <param name="headingMember">The name of a public field or property member of type T to be used as column name's in the pivoted DataTable object.</param>
/// <param name="membersHeading">The name of the column that lists the public fields and properties of type T.</param>        
static DataTable Pivot<T>(this IEnumerable<T> collection, string headingMember, string membersHeading)
{
    // get object information
    var type = typeof(T);
    var members = new List<MemberInfo>();
    members.AddRange(type.GetProperties(BindingFlags.Public | BindingFlags.Instance));
    members.AddRange(type.GetFields(BindingFlags.Public | BindingFlags.Instance));

    // create dataTable and establish schema
    var dt = new DataTable();
    dt.Columns.Add(membersHeading);
    foreach (var item in collection)
    {
        var member = members.Single(x => x.Name == headingMember);
        if (member is FieldInfo)                        
        {
            FieldInfo fieldInfo = (FieldInfo)member;
            dt.Columns.Add(fieldInfo.GetValue(item).ToString(), fieldInfo.FieldType);
        }

        if (member is PropertyInfo)
        {
            PropertyInfo propInfo = (PropertyInfo)member;
            dt.Columns.Add(propInfo.GetValue(item, null).ToString(), propInfo.PropertyType);
        }
    }

    // add rows to table
    foreach (MemberInfo member in members.Where(x => x.Name != headingMember))
    {
        var row = dt.NewRow();
        row[0] = member.Name;
        int i = 1;

        foreach (var item in collection)
        {
            if (member is FieldInfo)
                row[i++] = ((FieldInfo)member).GetValue(item);

            if (member is PropertyInfo)
                row[i++] = ((PropertyInfo)member).GetValue(item, null);
        }

        dt.Rows.Add(row);
    }

    return dt;
}

有没有有什么更好的方法来做到这一点?

【问题讨论】:

  • 这是很多问题的重复(搜索“pivot linq”)。
  • 我知道这是一个很老的问题,但你有没有找到“更好的方法”?
  • ¯_(ツ)_/¯ 不能说我记得需要“更好的方法”。查看我的旧代码,使用缓存的get 代表肯定可以提高性能。除此之外,如果您想要一个透视数据表,这对我来说看起来很合理。

标签: c# .net


【解决方案1】:

您可以通过在名称上使用 Group 来简单地旋转 IEnumerable。假设名称不同。转置数据后,您可以Convert IEnumerable to DataTable

【讨论】:

    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2012-02-24
    • 1970-01-01
    • 1970-01-01
    • 2010-12-06
    相关资源
    最近更新 更多