Я написал следующий метод.
public T GetByID(int id)
{
var dbcontext = DB;
var table = dbcontext.GetTable<T>();
return table.ToList().SingleOrDefault(e => Convert.ToInt16(e.GetType().GetProperties().First().GetValue(e, null)) == id);
}
В основном это метод в классе Generic, где T
- это класс в DataContext.
Метод получает таблицу из типа T (GetTable
) и проверяет для первого свойства (всегда являющегося идентификатором) введенный параметр.
Проблема с этим заключается в том, что мне пришлось сначала преобразовать таблицу элементов в список, чтобы выполнить GetType
в свойстве, но это не очень удобно, потому что все элементы таблицы должны быть перечислены и преобразованы в List
.
Как я могу реорганизовать этот метод, чтобы избежать ToList
на всей таблице?
[Обновление]
Причина, по которой я не могу выполнить Where
непосредственно в таблице, заключается в том, что я получаю это исключение:
Метод 'System.Reflection.PropertyInfo [] GetProperties()' не поддерживает перевод на SQL.
Потому что GetProperties
не может быть переведен на SQL.
[Обновление]
Некоторые люди предложили использовать интерфейс для T, но проблема в том, что параметр T
будет классом, который автоматически создается в [DataContextName].designer.cs, и поэтому я не могу заставить его реализовать интерфейс ( и не представляется возможным реализовать интерфейсы для всех этих "классов базы данных" LINQ, а также, файл будет восстановлен после добавления новых таблиц в DataContext, таким образом потеряв все записанные данные).
Итак, должен быть лучший способ сделать это...
[Обновление]
Теперь я реализовал свой код, например Neil Williams, но у меня все еще есть проблемы. Вот выдержки из кода:
Интерфейс:
public interface IHasID
{
int ID { get; set; }
}
DataContext [View Code]:
namespace MusicRepo_DataContext
{
partial class Artist : IHasID
{
public int ID
{
get { return ArtistID; }
set { throw new System.NotImplementedException(); }
}
}
}
Общий метод:
public class DBAccess<T> where T : class, IHasID,new()
{
public T GetByID(int id)
{
var dbcontext = DB;
var table = dbcontext.GetTable<T>();
return table.SingleOrDefault(e => e.ID.Equals(id));
}
}
Исключение выбрано в этой строке: return table.SingleOrDefault(e => e.ID.Equals(id));
и исключение:
System.NotSupportedException: The member 'MusicRepo_DataContext.IHasID.ID' has no supported translation to SQL.
[Обновить] Решение:
С помощью Denis Troller выложили ответ и ссылку на сообщение в Code Rant blog, мне наконец удалось найти решение:
public static PropertyInfo GetPrimaryKey(this Type entityType)
{
foreach (PropertyInfo property in entityType.GetProperties())
{
ColumnAttribute[] attributes = (ColumnAttribute[])property.GetCustomAttributes(typeof(ColumnAttribute), true);
if (attributes.Length == 1)
{
ColumnAttribute columnAttribute = attributes[0];
if (columnAttribute.IsPrimaryKey)
{
if (property.PropertyType != typeof(int))
{
throw new ApplicationException(string.Format("Primary key, '{0}', of type '{1}' is not int",
property.Name, entityType));
}
return property;
}
}
}
throw new ApplicationException(string.Format("No primary key defined for type {0}", entityType.Name));
}
public T GetByID(int id)
{
var dbcontext = DB;
var itemParameter = Expression.Parameter(typeof (T), "item");
var whereExpression = Expression.Lambda<Func<T, bool>>
(
Expression.Equal(
Expression.Property(
itemParameter,
typeof (T).GetPrimaryKey().Name
),
Expression.Constant(id)
),
new[] {itemParameter}
);
return dbcontext.GetTable<T>().Where(whereExpression).Single();
}