EF上的InvalidOperationException

时间:2014-09-17 05:26:46

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

我正在使用WPF MVVM Pettern和 我在SQL Server 2012上有一个简单的表,其ID(密钥)列在StoredProcedure中计算,称为PersonInsert :(这是简化的,但计算的内容比这更复杂,无论如何它是最后int

USE [Guard]
GO
SET ANSI_NULLS ON
GO
SET QUOTED_IDENTIFIER ON
GO

CREATE PROCEDURE [dbo].[PersonInsert]
@FName NVARCHAR(50) ,
@LName NVARCHAR(50) ,
@ID INT = NULL OUTPUT
AS
BEGIN
    BEGIN TRY
        SELECT  @ID = ISNULL(MAX(ID), 0) + 1
        FROM    dbo.Person


        INSERT  INTO [dbo].[Person]
                ( [ID] ,
                  [FName] ,
                  [LName] 
                )
        VALUES  ( @ID ,
                  @FName ,
                  @LName 
                )
    END TRY
    BEGIN CATCH
        DECLARE @ErrMsg NVARCHAR(MAX) ,
            @ErrSvr INT ,
            @ErrStt INT

        SELECT  @ErrMsg = ERROR_MESSAGE() + '|' + ERROR_PROCEDURE() + '|'
                + CAST(ERROR_LINE() AS NVARCHAR(5)) ,
                @ErrSvr = ERROR_SEVERITY() ,
                @ErrStt = ERROR_STATE()

        RAISERROR (@ErrMsg, @ErrSvr, @ErrStt)
    END CATCH
END
GO

在.net方面,我使用 EF 6.1 Code-First 处理数据并映射到SP,所以我有OnModelCreating这样:

modelBuilder.Entity(Of Person)().MapToStoredProcedures(
                    Sub(x)
                          x.Insert(Function(e) e.HasName("[dbo].[PersonInsert]"))
                          x.Update(Function(e) e.HasName("[dbo].[PersonUpdate]"))
                          x.Delete(Function(e) e.HasName("[dbo].[PersonDelete]"))
                    End Sub)

我的模特是:

<Table("Person")> _
Partial Public Class Person

    <DatabaseGenerated(DatabaseGeneratedOption.None), Key> _
    Public Property ID As Integer

    Public Property FName As String
    Public Property LName as String
End Class

现在奇怪的是,当我第一次尝试插入数据(添加新人)时,db.SaveChanged()效果很好,但是第二次它会抛出 InvalidOperation 异常消息:

  

对数据库的更改已提交成功,但更新对象上下文时发生错误。 ObjectContext可能处于不一致状态。内部异常消息:保存或接受更改失败,因为多个“Shoniz.Guard.WPFGuardApplication.Person”类型的实体具有相同的主键值。确保显式设置的主键值是唯一的。确保在数据库和Entity Framework模型中正确配置了数据库生成的主键。使用实体设计器进行数据库优先/模型优先配置。使用'HasDatabaseGeneratedOption'流畅API或' DatabaseGeneratedAttribute '进行代码优先配置。

他是对的!数据成功通过!我也是对的,因为我的ID(关键)列是正确计算的,而且它是完全唯一的。我在ID上使用 DatabaseGeneratedAttribute none 值,但结果是相同的:(

即使我重新查询数据库以检查重复键,我发现没有!

  1. 为什么会抛出此异常?
  2. 如何预防?
  3. 是否仍然在db.SaveChanges()之后忽略更改?

1 个答案:

答案 0 :(得分:2)

首先将DatabaseGeneratedOption.None更改为DatabaseGeneratedOption.Indetity 最后将存储过程更改为:

BEGIN TRY
        SELECT  @ID = ISNULL(MAX(ID), 0) + 1
        FROM    dbo.Person


        INSERT  INTO [dbo].[Person]
                ( [ID] ,
                  [FName] ,
                  [LName] 
                )
        VALUES  ( @ID ,
                  @FName ,
                  @LName 
                )
        --You have to tell EF this is returned Id of inserted person
        SELECT @ID as ID
    END TRY