Я пытаюсь создать общий метод с помощью EF4 для нахождения первичного ключа объекта.
пример
public string GetPrimaryKey<T>()
{
...
}
Для предоставления большего количества информации, я отделываюсь Tekpub StarterKit, и ниже класс, который я пытаюсь разбудить и работаю
using System;
using System.Collections.Generic;
using System.Linq;
using System.Web;
using System.Data.Objects;
using System.Data.Objects.ELinq;
using System.Data.Linq;
using Web.Infrastructure.Storage.EF4;
namespace Web.Infrastructure.Storage {
public class EFSession:ISession {
PuzzleEntities _db;//This is an ObjectContext
public EFSession() {
_db = new PuzzleEntities();
}
public void CommitChanges() {
_db.SaveChanges();
}
/// <summary>
/// Gets the table provided by the type T and returns for querying
/// </summary>
private ObjectSet<T> GetObjectSet<T>() where T:class {
return _db.CreateObjectSet<T>();
}
private T GetByPrimaryKey<T>() where T: class
{
.....
}
public void Delete<T>(System.Linq.Expressions.Expression<Func<T, bool>> expression) where T: class{
foreach (T item in All<T>().Where(expression))
{
GetObjectSet<T>().DeleteObject(item);
}
}
public void Delete<T>(T item) where T : class {
GetObjectSet<T>().DeleteObject(item);
}
public void DeleteAll<T>() where T : class {
foreach(T item in All<T>())
{
GetObjectSet<T>().DeleteObject(item);
}
}
public void Dispose() {
_db.Dispose();
}
public T Single<T>(System.Linq.Expressions.Expression<Func<T, bool>> expression) where T:class {
return GetObjectSet<T>().SingleOrDefault(expression);
}
public IQueryable<T> All<T>() where T : class {
return GetObjectSet<T>().AsQueryable();
}
public void Add<T>(T item) where T : class {
GetObjectSet<T>().AddObject(item);
}
public void Add<T>(IEnumerable<T> items) where T : class {
foreach (T item in items)
{
GetObjectSet<T>().AddObject(item);
}
}
public void Update<T>(T item) where T : class {
//nothing needed here
}
}
}
Итак, я наконец-то смог выяснить, как заставить это работать. Жаль, что я не потерял ссылку на блог, который я читал вчера вечером, так как я не писал код.
public T GetByPrimaryKey<T>(int id) where T : class
{
return (T)_db.GetObjectByKey(new EntityKey(_db.DefaultContainerName + "." + this.GetEntityName<T>(), GetPrimaryKeyInfo<T>().Name, id));
}
string GetEntityName<T>()
{
string name = typeof(T).Name;
if (name.ToLower() == "person")
return "People";
else if (name.Substring(name.Length - 1, 1).ToLower() == "y")
return name.Remove(name.Length - 1, 1) + "ies";
else if (name.Substring(name.Length - 1, 1).ToLower() == "s")
return name + "es";
else
return name + "s";
}
private PropertyInfo GetPrimaryKeyInfo<T>()
{
PropertyInfo[] properties = typeof(T).GetProperties();
foreach (PropertyInfo pI in properties)
{
System.Object[] attributes = pI.GetCustomAttributes(true);
foreach (object attribute in attributes)
{
if (attribute is EdmScalarPropertyAttribute)
{
if ((attribute as EdmScalarPropertyAttribute).EntityKeyProperty == true)
return pI;
}
else if (attribute is ColumnAttribute)
{
if ((attribute as ColumnAttribute).IsPrimaryKey == true)
return pI;
}
}
}
return null;
}
Надеюсь, это поможет кому-нибудь еще. Все, что я могу сказать, это то, что должно быть немного понятнее, как это сделать.
У каждой сущности EF4 есть свойство под названием EntityKey
, которое содержит массив EntityKeyValues
(массив присутствует в случае составного ключа).
Вы можете ссылаться на это непосредственно в своем экземпляре сущности или создать общий вспомогательный метод, который делает это скрытым образом. Если я смогу протестировать какой-нибудь образец кода, я опубликую его здесь.
Изменить : EntityKeyValue - это KeyValuePair
, где ключ
- это поле первичного ключа объекта, а значение
это связанное значение.
Например, у меня есть организация под названием Компания
, первичным ключом которой является поле , символ
.
var firstCompany = (from c in context.Companies select c).FirstOrDefault();
var kvp = firstCompany.EntityKey.EntityKeyValues[0];
// kvp shows {[Symbol, FOO]}
В моей песочнице я заметил, что это свойство было null
, когда я создавал объект в коде. Но как только я прочитал объект из базы данных, он был правильно заполнен. Таким образом, похоже, что концепция первичного ключа EF4 вступает в игру только после того, как он попадает в базу данных. Хотя вы можете установить его заранее, если хотите.