我正在创建将在我的应用程序中使用的默认并发策略。
我决定采用乐观策略。
我的所有实体都映射为Table per Type (TPT)
(使用继承)。我很快了解到,在 Entity Framework 上使用带有继承的 RowVersion 类型的列时会出现问题:
Product
Id INT IDENTITY PRIMARY KEY
RowVersion ROWVERSION
Car (inherits Product records)
Color TYNIINT NOT NULL,
AnotherProperty....
如果我更新 Car
表的记录,Product
表中的 RowVersion 列将不会更新。
我计划在 Product
中使用类型为 datetime2 (7)
的列,如果继承此表的表的任何记录被修改,则手动更新它。
我想我正在重新发明轮子。
在 Entity Framework 中使用 Table per Type (TPT)
时,是否有另一种方法可以通过 ROWVERSION
使用乐观并发策略?
编辑
我的映射:
class Product
{
int Id { get; set; }
string Name { get; set; }
byte[] RowVersion { get; set; }
}
class Car : Product
{
int Color { get; set; }
}
CodeFirst 约定。
只有 Product
实体的 RowVersion 属性具有自定义定义:
modelBuilder.Entity<Product>()
.Property(t => t.RowVersion)
.IsConcurrencyToken();
最佳答案
在 EF6 和 EF-core 中,当使用 SQL Server 时,你必须使用这个映射:
modelBuilder.Entity<Product>()
.Property(t => t.RowVersion)
.IsRowVersion(); // Not: IsConcurrencyToken
IsConcurrencyToken 确实将属性配置为并发 token ,但是(将其用于 byte[]
属性时)
- 数据类型是
varbinary(max)
- 如果你不初始化它,它的值总是
null
- 它的值不会在记录更新时自动递增。
IsRowVersion 另一方面,
- 具有数据类型
rowversion
(在 SQL Server 中,或在早期版本中为timestamp
),因此 - 它的值永远不会为空,并且
- 它的值总是在更新记录时自动递增。
- 它会自动将该属性配置为开放式并发 token 。
现在,当您更新 Car
时,您会看到两个更新语句:
DECLARE @p int
UPDATE [dbo].[Product]
SET @p = 0
WHERE (([Id] = @0) AND ([Rowversion] = @1))
SELECT [Rowversion]
FROM [dbo].[Product]
WHERE @@ROWCOUNT > 0 AND [Id] = @0
UPDATE [dbo].[Car]
SET ...
第一个语句不会更新任何内容,但它会增加 rowversion,如果 rowversion 在其间发生更改,它将抛出并发异常。
[System.ComponentModel.DataAnnotations.Schema.Timestamp]
属性是等效于 IsRowVersion()
的数据注释:
[Timestamp]
public byte[] RowVersion { get; set; }
请注意 official documentation是不正确的。它说 IsConcurrencyToken
是 [Timestamp]
属性的流畅等效项。但是,IsRowVersion
是等效的。
关于c# - 乐观并发 : IsConcurrencyToken and RowVersion,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/31330015/