是否可以在ObjectStateManagerChanged处理程序中修改实体对象?

时间:2011-08-23 17:19:06

标签: c# .net entity-framework entity-framework-4

这是我需要实现的业务逻辑:

数据模型中有多种实体类型,例如:User,UserProfile,UserProfileValue,UserExtendedData等。

在创建像User这样的实体时,我需要自动创建逻辑上链接的其他实体(通常在db中具有1:1的关系),例如UserProfile。

每个实体都有一个带OnCreating / OnCreated,OnUpdating / OnUpdated方法的控制器,这些方法是从我的datacontext类的重写的SaveChanges方法调用的。

public override int SaveChanges(SaveOptions options)
{
    IEnumerable<ObjectStateEntry> addedEntities = ObjectStateManager.GetObjectStateEntries(EntityState.Added);
    IEnumerable<ObjectStateEntry> changedEntities = ObjectStateManager.GetObjectStateEntries(EntityState.Modified);
    IEnumerable<ObjectStateEntry> deletedEntities = ObjectStateManager.GetObjectStateEntries(EntityState.Deleted);

    InvokeSavingControllers(addedEntities, changedEntities, deletedEntities);

    //some other code

    int rowsAffected = base.SaveChanges(options);

    InvokeSavedControllers(addedEntities, changedEntities, deletedEntities);

    return rowsAffected;
}

如果我在User.OnCreating()方法中实现自动创建,那么最终不会为所有自动创建的子实体调用OnCreating()控制器,因为集合addedEntities未使用新条目更新。

我的一个想法是通过处理ObjectStateManagerChanged数据上下文来执行从属实体的自动创建

private void ObjectStateManagerObjectStateManagerChanged(object sender, System.ComponentModel.CollectionChangeEventArgs e)
{
    if (e.Action == CollectionChangeAction.Add)
    {
        var state = ObjectStateManager.GetObjectStateEntry(e.Element).State;
        if (state == EntityState.Added)
        {
            //should call on creating handlers
            if (e.Element is User)
            {
                User user = (User)e.Element;
                user.UserProfile = UserProfile.CreateDefaultProfile();
            }
        }
    }
}

但不幸的是,如果我这样做,我会得到NullReferrence异常。即使我尝试在ObjectStateManagerObjectStateManagerChanged内修改用户的属性,我也会遇到异常。

有关如何实施所需功能的任何想法或建议?

修改

我得到的空参考:

System.NullReferenceException was unhandled by user code
  Message=Object reference not set to an instance of an object.
  Source=System.Data.Entity
  StackTrace:
       at System.Data.Objects.ObjectContext.AddSingleObject(EntitySet entitySet, IEntityWrapper wrappedEntity, String argumentName)
       at System.Data.Objects.ObjectContext.AddObject(String entitySetName, Object entity)
       at Edm.Entity.Entities.AddObject(Object entity) 

Edm.Entity.Entities.AddObject(Object entity)是将新的User对象添加到上下文中的位置。

EDIT2: 不知道我改变了什么但是上面的代码开始抛出以下异常:

<?xml version="1.0" encoding="utf-8" standalone="yes"?>
<error xmlns="http://schemas.microsoft.com/ado/2007/08/dataservices/metadata">
  <code></code>
  <message xml:lang="en-US">An error occurred while processing this request.</message>
  <innererror>
    <message>Object reference not set to an instance of an object.</message>
    <type>System.NullReferenceException</type>
    <stacktrace>   at System.Data.Objects.DataClasses.EntityReference`1.Exclude()&#xD;
   at System.Data.Objects.DataClasses.RelationshipManager.RemoveRelatedEntitiesFromObjectStateManager(IEntityWrapper wrappedEntity)&#xD;
   at System.Data.Objects.ObjectContext.AddObject(String entitySetName, Object entity)&#xD;
   at System.Data.Services.Providers.ObjectContextServiceProvider.CreateResource(String containerName, String fullTypeName)&#xD;
   at System.Data.Services.UpdatableWrapper.CreateResource(String containerName, String fullTypeName)&#xD;
   at System.Data.Services.Serializers.SyndicationDeserializer.CreateObject(SegmentInfo segmentInfo, Boolean topLevel, SyndicationItem item)&#xD;
   at System.Data.Services.Serializers.SyndicationDeserializer.CreateSingleObject(SegmentInfo segmentInfo)&#xD;
   at System.Data.Services.Serializers.Deserializer.ReadEntity(RequestDescription requestDescription)&#xD;
   at System.Data.Services.Serializers.Deserializer.HandlePostRequest(RequestDescription requestDescription)&#xD;
   at System.Data.Services.DataService`1.HandlePostOperation(RequestDescription description, IDataService dataService)&#xD;
   at System.Data.Services.DataService`1.ProcessIncomingRequest(RequestDescription description, IDataService dataService)&#xD;
   at System.Data.Services.DataService`1.BatchDataService.HandleBatchContent(Stream responseStream)</stacktrace>
  </innererror>
</error>

1 个答案:

答案 0 :(得分:2)

我有完全相同的问题。在ObjectStateManagerChanged事件期间添加另一个EntityObject似乎是一个问题。奇怪的是我的代码(以及你的代码可能也是如此)在.net 3.5中运行良好,但在4.0中给出了NullReferenceException。

我调试了.net框架,并发现它在调用ObjectStateManagerChanged事件后在内部ObjectContext.AddSingleObject方法中崩溃。发生此异常是因为内部ProcessedEntities散列集在公共AddObject方法的末尾变为null。但是如果在ObjectStateManagerChanged事件期间创建了EntityObject,则不应将此hashset设为null。

这似乎是一个.net framework 4.0 bug。