案例—–< CaseSubjectRelationships> —— CaseSubjects
更充分地说
案例(ID,CaseTypeID,…….)
CaseSubjects(ID,DisplayName,CRMSPIN)
CaseSubjectsRelationships(CaseID,SubjectID,PrimarySubject,RelationToCase,…)
在我的多对多链接表中是与主题与具体情况相关联的附加属性 – 例如开始日期,结束日期,自由文本关系(例如观察者,创建者等)
已创建实体框架数据模型 – ASP.NET 4.0版
我有一个名为CreateNewCase的方法的WCF服务,它接受一个Case对象(由Entity Framework创建的实体)作为其参数 – 它的工作是将案例保存到数据库中。
WCF服务由第三方工具调用。以下是发送的SOAP:
<s:Envelope xmlns:s="http://schemas.xmlsoap.org/soap/envelope/"> <s:Body> <CreateNewCase xmlns="http://tempuri.org/"> <c xmlns:a="http://schemas.datacontract.org/2004/07/CAMSModel"> <a:CaseSubjectsRelationships> <a:CaseSubjectsRelationship> <a:CaseSubject> <a:CRMSPIN>601</a:CRMSPIN> <a:DisplayName>Fred Flintstone</a:DisplayName> </a:CaseSubject> <a:PrimarySubject>true</a:PrimarySubject> <a:RelationToCase>Interested</a:RelationToCase> <a:StartDate>2011-07-12T00:00:00</a:StartDate> </a:CaseSubjectsRelationship> <a:CaseSubjectsRelationship> <a:CaseSubject> <a:CRMSPIN>602</a:CRMSPIN> <a:DisplayName>Barney Rubble</a:DisplayName> </a:CaseSubject> <a:RelationToCase>Observer</a:RelationToCase> <a:StartDate>2011-07-12T00:00:00</a:StartDate> </a:CaseSubjectsRelationship> </a:CaseSubjectsRelationships> <a:CaseType> <a:Identifier>Change of Occupier</a:Identifier> </a:CaseType> <a:Description>Case description</a:Description> <a:Priority>5</a:Priority> <a:QueueIdentifier>Queue One</a:QueueIdentifier> <a:Title>Case title</a:Title> </c> </CreateNewCase> </s:Body> </s:Envelope>
WCF引擎将它反序列化为一个Case实体,我正确地,当我看到调试器中的一切都正确设置。
我想做的只是创建一个新的CaseSubject,如果数据库中没有指定CRMSPIN的条目(CRMSPIN是中央客户数据库的引用号)
所以,在下面的例子中,我想看看我在CaseSubjects中是否已经有一个CRMSPIN 601的一个条目,如果我这样做,我不想创建另一个(重复的)条目,而是使新的case链接到现在的主题(虽然新行将需要在CaseSubjectsRelationships中创建具体的“附加”信息,如关系等)
这是我试图这样做的.NET代码。
Public Class CamsService Implements ICamsService Public Function CreateNewCase(c As CAMSModel.Case) As String Implements ICamsService.CreateNewCase Using ctx As New CAMSEntities ' Find the case type ' Dim ct = ctx.CaseTypes.SingleOrDefault(Function(x) x.Identifier.ToUpper = c.CaseType.Identifier.ToUpper) ' Give an error if no such case type ' If ct Is Nothing Then Throw New CaseTypeInvalidException(String.Format("The case type {0} is not valid.",c.CaseType.Identifier.ToString)) End If ' Set the case type based on that found in database: ' c.CaseType = ct For Each csr In c.CaseSubjectsRelationships Dim spin As String = csr.CaseSubject.CRMSPIN Dim s As CaseSubject = ctx.CaseSubjects.SingleOrDefault(Function(x) x.CRMSPIN = spin) If Not s Is Nothing Then ' The subject has been found based on CRMSPIN so set the subject in the relationship ' csr.CaseSubject = s End If Next c.CreationChannel = "Web service" c.CreationDate = Now.Date ' Save it ' ctx.AddToCases(c) ctx.SaveChanges() End Using ' Return the case reference ' Return c.ID.ToString End Function End Class
正如你可以看到的,而不是For Each循环,我尝试根据CRMSPIN获取主题,如果我得到一些东西,那么我更新“CaseSubject”实体。 (我也尝试过csr.SubjectID = s.ID,而不是设置整个实体,并且我已经尝试设置它们!
然而,即使在ctx.SaveChanges()行上放置断点,并且查看主题的设置以及在调试器中看到它看起来不错,它总是在CaseSubjects表中创建一个新行。
我原则上可以看到这应该是正常的 – 你会看到我已经完成了相同的事情为Case类型 – 我已经选择了XML中发送的标识符,通过上下文找到具有该标识符的实体,然后更改了该案例。 CaseType到我发现的实体。当它保存时,它的工作原理和预期,没有重复的行。
我只是试图将同样的理论应用于多对多关系的一边。
以下是.edmx的一些(希望相关的)提取
<EntitySet Name="Cases" EntityType="CAMSModel.Store.Cases" store:Type="Tables" Schema="dbo" /> <EntitySet Name="CaseSubjects" EntityType="CAMSModel.Store.CaseSubjects" store:Type="Tables" Schema="dbo" /> <EntitySet Name="CaseSubjectsRelationships" EntityType="CAMSModel.Store.CaseSubjectsRelationships" store:Type="Tables" Schema="dbo" /> <AssociationSet Name="FK_CaseSubjectsRelationships_Cases" Association="CAMSModel.Store.FK_CaseSubjectsRelationships_Cases"> <End Role="Cases" EntitySet="Cases" /> <End Role="CaseSubjectsRelationships" EntitySet="CaseSubjectsRelationships" /> </AssociationSet> <AssociationSet Name="FK_CaseSubjectsRelationships_CaseSubjects" Association="CAMSModel.Store.FK_CaseSubjectsRelationships_CaseSubjects"> <End Role="CaseSubjects" EntitySet="CaseSubjects" /> <End Role="CaseSubjectsRelationships" EntitySet="CaseSubjectsRelationships" /> </AssociationSet>
编辑:CaseSubjectsRelationships对象的CaseSubject属性的属性设置器:
/// <summary> /// No Metadata Documentation available. /// </summary> <XmlIgnoreAttribute()> <SoapIgnoreAttribute()> <DataMemberAttribute()> <EdmRelationshipNavigationPropertyAttribute("CAMSModel","FK_CaseSubjectsRelationships_CaseSubjects","CaseSubject")> Public Property CaseSubject() As CaseSubject Get Return CType(Me,IEntityWithRelationships).RelationshipManager.GetRelatedReference(Of CaseSubject)("CAMSModel.FK_CaseSubjectsRelationships_CaseSubjects","CaseSubject").Value End Get Set CType(Me,"CaseSubject").Value = value End Set End Property
解决方法
所以,基本上这就是我认为正在发生的事情。
在您的代码中,当您从上下文中获取某些内容时:
Dim ct = ctx.CaseTypes.SingleOrDefault(Function(x) x.Identifier.ToUpper = c.CaseType.Identifier.ToUpper)
这个ct在上下文中。您从服务(c)中反序列化的方法参数不在上下文中。您可以将上下文视为“对象跟踪和提取”实体,这样可以确保附件中的所有内容都可以了解任何更改(如果是新的,已删除的)。
所以,当你到达这个部分:
' Set the case type based on that found in database: ' c.CaseType = ct
在分配一些附加到不附加的东西的时刻,未连接的对象也将被拉入上下文中 – 不能有“部分”附加的实体 – 如果附加的实体,它引用的所有内容也必须被附加。所以,这是c被“拖”到上下文(隐含的)的时刻。当它进入上下文时,它将被标记为“新”,因为它不知道任何东西(它不知道它,没有更改跟踪信息…)。
所以,现在关于该对象c的所有内容都是在上下文中,当您查询上下文时:
Dim s As CaseSubject = ctx.CaseSubjects.SingleOrDefault(Function(x) x.CRMSPIN = spin)
它会显示,确实有一个对象与CRMSPIN,它已经附加 – “嘿,没有必要去数据库,我已经有了! (试图聪明,避免db命中),它将返回您自己的对象。
最后,当您保存所有内容时,它将被保存,但是将附加的c和标记为“new”的所有子对象都将被插入,而不是更新。
最简单的修复将是首先从上下文查询所需的内容,然后才开始将其分配给对象的属性。另外,看看UpdateCurrentValues,也可能有帮助…