c# - 乐观并发 : IsConcurrencyToken and RowVersion

标签 c# sql-server entity-framework concurrency ef-code-first

我正在创建将在我的应用程序中使用的默认并发策略。

我决定采用乐观策略。

我的所有实体都映射为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/

相关文章:

c# - IronPython 覆盖内置方法

C#:如何从现有基类对象构造派生类?

c# - ListView的集合中多个IsSelected

c# - 过滤but属性和子实体属性

entity-framework - Entity Framework 在提交时全部删除

c# - 为什么 Sqrt(-4) 是 NaN 而 Sqrt(1/-4)=0?

sql-server - SQL 始终加密与 SQL 动态数据屏蔽

sql - 从 SQL Server 表中选择具有最大日期的不同行?

sql - 当 SQL 中的值为空时,如何在 INSERT 上跳过列条目?

mysql - ADO.NET linq 到 MYSQL,orderby 不工作