Преобразовать DataTable в общий список в C # - PullRequest
25 голосов
/ 05 ноября 2010

Отказ от ответственности: я знаю, что его спросили во многих местах в SO.
Мой запрос немного другой.

Язык кодирования: C # 3.5

У меня есть DataTable с именем cardsTable, который извлекает данные из БД, и у меня есть класс Cards, у которого есть только некоторые свойства (без конструктора)

public class Cards
{
    public Int64 CardID { get; set; }
    public string CardName { get; set; }
    public Int64 ProjectID { get; set; }
    public Double CardWidth { get; set; }
    public Double CardHeight { get; set; }
    public string Orientation { get; set; }
    public string BackgroundImage { get; set; }
    public string Background { get; set; }
}

Я хочу вставить данные cardsTable в объект типа List.
В моих данных будут нулевые поля, поэтому метод не должен давать ошибку при преобразовании данных. Является ли приведенный ниже метод лучшим способом?

DataTable dt = GetDataFromDB();
List<Cards> target = dt.AsEnumerable().ToList().ConvertAll(x => new Cards { CardID = (Int64)x.ItemArray[0] });

Ответы [ 9 ]

34 голосов
/ 05 ноября 2010

Вы могли бы значительно сократить его. Вы можете думать о методе расширения Select() как о преобразователе типов. Тогда преобразование можно записать так:

List<Cards> target = dt.AsEnumerable()
    .Select(row => new Cards
    {
        // assuming column 0's type is Nullable<long>
        CardID = row.Field<long?>(0).GetValueOrDefault(),
        CardName = String.IsNullOrEmpty(row.Field<string>(1))
            ? "not found"
            : row.Field<string>(1),
    }).ToList();
11 голосов
/ 05 ноября 2010

Я думаю, что все решения можно улучшить и сделать метод более общим, если вы используете некоторые соглашения и рефлексию. Допустим, вы называете свои столбцы в таблице данных тем же именем, что и свойства в вашем объекте, тогда вы можете написать что-то, что будет проверять все ваши свойства вашего объекта, а затем искать этот столбец в таблице данных для сопоставления значения.

Я сделал обратное, то есть ... из IList в datatable, и код, который я написал, можно увидеть по адресу: http://blog.tomasjansson.com/convert-datatable-to-generic-list-extension/

Не должно быть так сложно идти другим путем, и должно быть так сложно перегружать функции, чтобы вы могли предоставить информацию о том, какие свойства вы хотите включить или исключить.

EDIT: Итак, код, чтобы заставить его работать:

public static class DataTableExtensions
{
    private static Dictionary<Type,IList<PropertyInfo>> typeDictionary = new Dictionary<Type, IList<PropertyInfo>>();
    public static IList<PropertyInfo> GetPropertiesForType<T>()
    {
        var type = typeof(T);
        if(!typeDictionary.ContainsKey(typeof(T)))
        {
            typeDictionary.Add(type, type.GetProperties().ToList());
        }
        return typeDictionary[type];
    }

    public static IList<T> ToList<T>(this DataTable table) where T : new()
    {
        IList<PropertyInfo> properties = GetPropertiesForType<T>();
        IList<T> result = new List<T>();

        foreach (var row in table.Rows)
        {
            var item = CreateItemFromRow<T>((DataRow)row, properties);
            result.Add(item);
        }

        return result;
    }

    private static T CreateItemFromRow<T>(DataRow row, IList<PropertyInfo> properties) where T : new()
    {
        T item = new T();
        foreach (var property in properties)
        {
            property.SetValue(item, row[property.Name], null);
        }
        return item;
    }

}

Если у вас есть DataTable, вы можете просто написать yourTable.ToList<YourType>(), и он создаст список для вас. Если у вас более сложный тип с вложенными объектами, вам нужно обновить код. Одно из предложений - просто перегрузить метод ToList, чтобы принять params string[] excludeProperties, который содержит все ваши свойства, которые не должны отображаться. Конечно, вы можете добавить проверку нуля в цикл foreach метода CreateItemForRow.

ОБНОВЛЕНИЕ: Добавлен статический словарь для сохранения результата от операции отражения, чтобы сделать его немного быстрее. Я не скомпилировал код, но он должен работать :).

6 голосов
/ 06 декабря 2013

Просто небольшое упрощение.Я не использую ItemArray:

List<Person> list = tbl.AsEnumerable().Select(x => new Person
                    {
                        Id = (Int32) (x["Id"]),
                        Name = (string) (x["Name"] ?? ""),
                        LastName = (string) (x["LastName"] ?? "")
                    }).ToList();
6 голосов
/ 05 ноября 2010

.ToList () находится не в том месте, и если некоторые поля могут быть нулевыми, вам придется иметь дело с ними, так как они не преобразуются в Int64, если они нулевые

DataTable dt = GetDataFromDB();
List<Cards> target = dt.AsEnumerable().Select(
  x => new Cards { CardID = (Int64)(x.ItemArray[0] ?? 0) }).ToList();
2 голосов
/ 05 ноября 2010

хорошо, это решение в одну строку

это зависит от того, действительно ли вы знаете все данные в базе данных действительны и не будут содержать ничего, что нарушит вышеприведенное

например, обнуляемое поле, когда вы этого не ожидаете - возможно, из-за левого соединения в eh sql, которое генерирует данные.

Так что, если вы проверили данные раньше, тогда да - я был с готов предложить несколькоlinq - но вы получили tht.

Если вам нужна некоторая проверка, однако вам, вероятно, следует просто пройтись по даторам, сгенерировать ваш объект, как указано выше, и добавить его в коллекцию ... это также позволит вам обрабатыватьошибки в одной строке и все еще обрабатывают остальные.

Вот так я все равно это вижу

(чёрт, я пришел понизить что-то, поэтому мой представитель был 1024)

0 голосов
/ 19 сентября 2018

Вот простой способ преобразования в общий список в c # с условием Where

List<Filter> filter = ds.Tables[0].AsEnumerable()
                        .Where(x => x.Field<int>("FilterID") == 5)
                        .Select(row => new Filter
                        {
                            FilterID = row.Field<int>("FilterID"),
                            FilterName = row.Field<string>("FilterName")
                        }).ToList();

Сначала определите свойства и используйте согласно

public class Filter
{
    public int FilterID { get; set; }
    public string FilterName { get; set; }
}

Put Package:

using System.Linq;
using System.Collections.Generic;
0 голосов
/ 28 июня 2017

Опаздывает, но это может быть полезно. Можно вызвать с помощью:

table.Map (); или вызвать Func для фильтрации значений.

Вы даже можете изменить имя отображения между свойством type и заголовком DataColumn, установив атрибуты для свойства.

[AttributeUsage(AttributeTargets.Property)]
    public class SimppleMapperAttribute: Attribute
    {
        public string HeaderName { get; set; }
    }


     public static class SimpleMapper
{
    #region properties
    public static bool UseDeferredExecution { get; set; } = true;
    #endregion

#region public_interface  


    public static IEnumerable<T> MapWhere<T>(this DataTable table, Func<T, bool> sortExpression) where T:new()
    {
        var result = table.Select().Select(row => ConvertRow<T>(row, table.Columns, typeof(T).GetProperties())).Where((t)=>sortExpression(t));
        return UseDeferredExecution ? result : result.ToArray();
    }
    public static IEnumerable<T> Map<T>(this DataTable table) where T : new()
    {
        var result = table.Select().Select(row => ConvertRow<T>(row, table.Columns, typeof(T).GetProperties()));
        return UseDeferredExecution ? result : result.ToArray();
    }
    #endregion

#region implementation_details
    private static T ConvertRow<T>(DataRow row, DataColumnCollection columns, System.Reflection.PropertyInfo[] p_info) where T : new()
    {
        var instance = new T();
        foreach (var info in p_info)
        {
            if (columns.Contains(GetMappingName(info))) SetProperty(row, instance, info);             
        }
        return instance;
    }

    private static void SetProperty<T>(DataRow row, T instance, System.Reflection.PropertyInfo info) where T : new()
    {
        string mp_name = GetMappingName(info);
        object value = row[mp_name];
        info.SetValue(instance, value);
    }

    private static string GetMappingName(System.Reflection.PropertyInfo info)
    {
        SimppleMapperAttribute attribute = info.GetCustomAttributes(typeof(SimppleMapperAttribute),true).Select((o) => o as SimppleMapperAttribute).FirstOrDefault();
        return attribute == null ? info.Name : attribute.HeaderName;
    }
#endregion
}
0 голосов
/ 19 октября 2015

Я основывался на логике Томаса Янссона, чтобы включить атрибут «Игнорировать». Это позволяет мне добавлять другие атрибуты в загружаемый класс, не прерывая саму загрузку DataTable-To-Class.

В качестве альтернативы я также рассмотрел возможность добавления отдельного параметра, который содержит фактическое имя столбца для чтения в DataTable. В этом случае вместо использования «row [property.Name]» вы будете использовать row [attribute.Name] »или что-то подобное для этого конкретного свойства.

public static class DataTableExtensions
{
    [AttributeUsage(AttributeTargets.Property, Inherited = false, AllowMultiple = true)]
    public sealed class IgnoreAttribute : Attribute { public IgnoreAttribute() { } }

    private static Dictionary<Type, IList<PropertyInfo>> typeDictionary = new Dictionary<Type, IList<PropertyInfo>>();

    public static IList<PropertyInfo> GetPropertiesForType<T>()
    {
        var type = typeof(T);

        if (!typeDictionary.ContainsKey(typeof(T)))
            typeDictionary.Add(type, type.GetProperties().ToList());

        return typeDictionary[type];
    }

    public static IList<T> ToList<T>(this DataTable table) where T : new()
    {
        IList<PropertyInfo> properties = GetPropertiesForType<T>();
        IList<T> result = new List<T>();

        foreach (var row in table.Rows)
            result.Add(CreateItemFromRow<T>((DataRow)row, properties));

        return result;
    }

    private static T CreateItemFromRow<T>(DataRow row, IList<PropertyInfo> properties) where T : new()
    {
        T item = new T();

        foreach (var property in properties)
        {
            // Only load those attributes NOT tagged with the Ignore Attribute
            var atr = property.GetCustomAttribute(typeof(IgnoreAttribute));
            if (atr == null)
                property.SetValue(item, row[property.Name], null);
        }

        return item;
    }
}
0 голосов
/ 17 октября 2015

Вы можете сопоставить таблицу данных с классом модели, используя общий класс, как показано ниже.

Общий класс

 public static class DataTableMappingtoModel
    {
        /// <summary>
        /// Maps Data Table values to coresponded model propertise
        /// </summary>
        /// <typeparam name="T"></typeparam>
        /// <param name="dt"></param>
        /// <returns></returns>
        public static List<T> MappingToEntity<T>(this DataTable dt) 
        {
            try
            {
                var lst = new List<T>();
                var tClass = typeof (T);
                PropertyInfo[] proInModel = tClass.GetProperties();
                List<DataColumn> proInDataColumns = dt.Columns.Cast<DataColumn>().ToList();
                T cn;
                foreach (DataRow item in dt.Rows)
                {
                    cn = (T) Activator.CreateInstance(tClass);
                    foreach (var pc in proInModel)
                    {


                            var d = proInDataColumns.Find(c => string.Equals(c.ColumnName.ToLower().Trim(), pc.Name.ToLower().Trim(), StringComparison.CurrentCultureIgnoreCase));
                            if (d != null)
                                pc.SetValue(cn, item[pc.Name], null);


                    }
                    lst.Add(cn);
                }
                return lst;
            }
            catch (Exception e)
            {
                throw e;
            }
        }
    }

Класс модели

public class Item
{
    public string ItemCode { get; set; }
    public string Cost { get; set; }
    public override string ToString()
    {
        return "ItemCode : " + ItemCode + ", Cost : " + Cost;
    }
}

Создать таблицу данных

public DataTable getTable()
{
    DataTable dt = new DataTable();
    dt.Columns.Add(new DataColumn("ItemCode", typeof(string)));
    dt.Columns.Add(new DataColumn("Cost", typeof(string)));
    DataRow dr;
    for (int i = 0; i < 10; i++)
    {
        dr = dt.NewRow();
        dr[0] = "ItemCode" + (i + 1);
        dr[1] = "Cost" + (i + 1);
        dt.Rows.Add(dr);
    }
    return dt;
}

Теперь мы можем преобразовать эту таблицу данных в список, как показано ниже:

DataTable dt = getTable();
List<Item> lst = dt.ToCollection<Item>();
foreach (Item cn in lst)
{
    Response.Write(cn.ToString() + "<BR/>");
}

Надеюсь, вам поможет

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...