我正在使用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 值,但结果是相同的:(
即使我重新查询数据库以检查重复键,我发现没有!
db.SaveChanges()
之后忽略更改? 答案 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