我正在为我的 EntityFramework 对象创建扩展,如如何:自定义生成的数据对象中所述,但在其中一些扩展中,我需要获取实例的 ObjectContext 以查找模型中的其他一些值。我找到了提示 24 – 如何从实体获取 ObjectContext,但这是几年前写的,在类似的 SO 问题中引用了它,但我真的希望现在有更好的答案。
当然,这必须是经常需要的东西,以至于从实体本身检索实体的对象上下文应该得到官方方法的支持。
提前感谢您提供有关此实施的最新信息。
我正在为我的 EntityFramework 对象创建扩展,如如何:自定义生成的数据对象中所述,但在其中一些扩展中,我需要获取实例的 ObjectContext 以查找模型中的其他一些值。我找到了提示 24 – 如何从实体获取 ObjectContext,但这是几年前写的,在类似的 SO 问题中引用了它,但我真的希望现在有更好的答案。
当然,这必须是经常需要的东西,以至于从实体本身检索实体的对象上下文应该得到官方方法的支持。
提前感谢您提供有关此实施的最新信息。
还有另一种解决方案,使用连接的属性。
使用连接属性看起来像这样(警告:未经测试的代码):
public partial class Database1Entities
{
private struct ObjectContextProperty { }
partial void OnContextCreated()
{
this.ObjectMaterialized += (_, e) =>
{
e.Entity.GetConnectedProperty<Database1Entities, ObjectContextProperty>().Set(this);
};
this.ObjectStateManager.ObjectStateManagerChanged += (_, e) =>
{
if (e.Action == CollectionChangeAction.Add)
{
e.Element.GetConnectedProperty<Database1Entities, ObjectContextProperty>().Set(this);
}
else if (e.Action == CollectionChangeAction.Remove)
{
e.Element.GetConnectedProperty<Database1Entities, ObjectContextProperty>().Set(null);
}
};
}
/// <summary>
/// Gets the object context for the entity. Returns <c>null</c> if the entity is detached.
/// </summary>
/// <param name="entity">The entity for which to return the object context.</param>
public static Database1Entities FromEntity(EntityObject entity)
{
return entity.GetConnectedProperty<Database1Entities, ObjectContextProperty>().GetOrConnect(null);
}
}
然后您可以使用Database1Entities.FromEntity
从实体对象中获取对象上下文。如果需要,您还可以在实体对象上定义实际属性:
public partial class Table1
{
/// <summary>
/// Gets the object context for this entity. Returns <c>null</c> if the entity is detached.
/// </summary>
public Database1Entities ObjectContext { get { return Database1Entities.FromEntity(this); } }
}
在此解决方案中,ObjectContext
实体对象的属性是可选的。
不,没有任何这样的方法。所描述的解决方法看起来是唯一的选择,因为实体派生自EntityObject
定义为:
[Serializable, DataContract(IsReference=true)]
public abstract class EntityObject : StructuralObject, IEntityWithKey,
IEntityWithChangeTracker, IEntityWithRelationships
{
...
}
据我所知,只会IEntityWithRelationships.RelationshipManager
导致ObjectContext
. 这在 EF 4 中没有改变。
此外,从实体访问上下文也不是很常见。我可以想象这对于在 EF 之上实现 Active Record 模式会有所帮助,但在这种情况下,您可能还可以控制在实体的静态方法内创建上下文,因此您应该能够将其设置为实体。在其他情况下,我会说这是您应该尽可能避免的事情。
这就是我使用的;这是一种基于约定的方法,很容易添加到项目中。
首先,将钩子添加到您的对象上下文中:
public partial class Database1Entities
{
partial void OnContextCreated()
{
this.ObjectMaterialized += (_, e) =>
{
try
{
dynamic entity = e.Entity;
entity.ObjectContext = this;
}
catch (RuntimeBinderException)
{
}
};
this.ObjectStateManager.ObjectStateManagerChanged += (_, e) =>
{
if (e.Action == CollectionChangeAction.Add)
{
try
{
dynamic entity = e.Element;
entity.ObjectContext = this;
}
catch (RuntimeBinderException)
{
}
}
else if (e.Action == CollectionChangeAction.Remove)
{
try
{
dynamic entity = e.Element;
entity.ObjectContext = null;
}
catch (RuntimeBinderException)
{
}
}
};
}
}
这将尝试动态设置ObjectContext
在与对象上下文关联的任何实体上调用的属性。
接下来,ObjectContext
向实体类型添加一个:
public partial class Table1
{
/// <summary>
/// Gets or sets the context for this entity.
/// This should not be set by end-user code; this property will be set
/// automatically as entities are created or added,
/// and will be set to <c>null</c> as entities are detached.
/// </summary>
public Database1Entities ObjectContext { get; set; }
}
此解决方案确实需要ObjectContext
为每个实体类型添加一个属性。