代码之家  ›  专栏  ›  技术社区  ›  Udontknow

实体框架:RowVersion值为空

  •  1
  • Udontknow  · 技术社区  · 6 年前

    我正在使用EntityFramework6.2.0和本地MSSQL(MDF)数据库。

    我有几个类型都是从我的主类型“Entity”派生而来的(使用了“每类型表”策略)。现在,我正在尝试实现乐观锁定。

    在我的EDMX文件中,我将属性RowVersion添加到Entity(SQL-DB中为8字节的固定长度字节数组:“[RowVersion]binary(8)NOT NULL”),并将该proepty的并发模式设置为“fixed”。我用“Timestamp”属性标记实体类中的属性:

    [System.ComponentModel.DataAnnotations.Schema.Table("EntitySet", Schema = "RightsManager")]
    public partial class Entity
    {
        public int Id { get; set; }
        public string Name { get; set; }
        public System.DateTime ActiveFrom { get; set; }
        public Nullable<System.DateTime> ActiveUntil { get; set; }
    
        [System.ComponentModel.DataAnnotations.Timestamp]
        public byte[] RowVersion { get; set; }
    }
    

    我还将代码添加到DBContext子体的OnModelCreating中,以指示要使用的RowVersion:

        protected override void OnModelCreating(DbModelBuilder modelBuilder)
        {
            Database.SetInitializer<RightsManagerContext>(null);            
            base.OnModelCreating(modelBuilder);
            modelBuilder.Entity<Entity>().Property(p => p.RowVersion).IsRowVersion();
            modelBuilder.Entity<Product>().Property(p => p.RowVersion).IsRowVersion();
        }
    

    问题:在插入新产品时,会抛出一个SQL错误。这是我使用的单元测试:

        [TestMethod]
        public void TestCreateProduct()
        {
            using (var context = GetContext())
            {
                var newProduct = new Product
                {
                    Name = "New product",
                    ActiveFrom = DateTime.Now
    
                };
                context.Entry(newProduct).State = System.Data.Entity.EntityState.Added;
                var objectsWritten = context.SaveChanges();
                Assert.AreNotEqual(0, objectsWritten);
            };
        }
    

    引发了最内部的异常:

    显然,EF不是自动填充值,而是像处理其他字段一样处理字段。我错过了什么?

    1 回复  |  直到 6 年前
        1
  •  2
  •   Udontknow    6 年前

    我想我误解了IsRowVersion/Timestamp是一个与数据库无关的问题。似乎只有在创建表时使用MSSQL特定的数据库字段类型“rowversion”时,整个机制才有效。所有其他数据库(如Oracle、DB2等)都不在范围内。

    由于我试图在我的项目中保持DBMS中立性,我将不得不用“IsConcurrencyToken”手动实现这样一个特性。