5

我正在为我的 EntityFramework 对象创建扩展,如如何:自定义生成的数据对象中所述,但在其中一些扩展中,我需要获取实例的 ObjectContext 以查找模型中的其他一些值。我找到了提示 24 – 如何从实体获取 ObjectContext,但这是几年前写的,在类似的 SO 问题中引用了它,但我真的希望现在有更好的答案。

当然,这必须是经常需要的东西,以至于从实体本身检索实体的对象上下文应该得到官方方法的支持。

提前感谢您提供有关此实施的最新信息。

4

3 回答 3

3

还有另一种解决方案,使用连接的属性

使用连接属性看起来像这样(警告:未经测试的代码):

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实体对象的属性是可选的。

于 2011-04-22T16:07:17.863 回答
0

不,没有任何这样的方法。所描述的解决方法看起来是唯一的选择,因为实体派生自EntityObject定义为:

[Serializable, DataContract(IsReference=true)]
public abstract class EntityObject : StructuralObject, IEntityWithKey,  
    IEntityWithChangeTracker, IEntityWithRelationships
{
    ...
}

据我所知,只会IEntityWithRelationships.RelationshipManager导致ObjectContext. 这在 EF 4 中没有改变。

此外,从实体访问上下文也不是很常见。我可以想象这对于在 EF 之上实现 Active Record 模式会有所帮助,但在这种情况下,您可能还可以控制在实体的静态方法内创建上下文,因此您应该能够将其设置为实体。在其他情况下,我会说这是您应该尽可能避免的事情。

于 2011-04-18T19:38:34.230 回答
0

这就是我使用的;这是一种基于约定的方法,很容易添加到项目中。

首先,将钩子添加到您的对象上下文中:

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为每个实体类型添加一个属性。

于 2011-04-22T16:05:35.023 回答