首页 > 解决方案 > 自动递增数字的最佳实践 EF Core

问题描述

我正在寻找最佳实践,以便使用 EF Core 在我的数据库中增加一些数字。

我有一个名为 PARENT 的数据库模型。在我的应用程序中,我可以有多个 PARENTS。每个家长都有几个孩子。

家长 1:孩子 1 - 孩子 2

父母 2:孩子 1 - 孩子 2 - 孩子 3

我希望在每个 PARENT 内部,每个孩子都按列递增编号:int Number。这表示,

父母1.孩子1 = 001;

父母1.孩子2 = 002;

PARENT2.CHILD1 = 001;

父母2.孩子2 = 002;

...

所以,它必须是唯一的,并且由父级递增..

谁能给我一些最佳实践,以便以有效的方式实现这一目标?

更新

public class Bar {
    public Guid BarId {get;set;}
    public ICollection<Client> Clients {get;set;}
}

public class Client {
    public Guid ClientId {get;set;}
    public Guid BarId {get;set;}
    public int ClientNumber {get;set;}
}

public class Context : DBContext {
    entity.HasKey(e => new { e.BarId, e.ClientId }).HasName("PK_CLIENT");
                entity.Property(e => e.ClientId).ValueGeneratedWhenDefault();
    entity.Property(e => e.ClientNumber).ValueGeneratedOnAdd();
    entity.HasIndex(e => new {e.BarId, e.ClientNumber}).IsUnique()
        .HasName("IX_CLIENT_NUMBER");
}

通过调用ValueGeneratedOnAdd(),我使数字列递增,但我希望它为每个 Bar 递增,因此它应该允许不同 bar 之间的重复客户编号,但不能在同一个 Bar 上,并且还按 Bar 递增它们,而不仅仅是通过 Client 实体.

标签: c#asp.net-coreentity-framework-coreautomapper

解决方案


一种方法是执行以下操作;

1) 使用整数作为键,并将 ClientNumber 属性设置为可为空。

    public class Bar
    {
        public int BarId { get; set; }
        public ICollection<Client> Clients { get; set; }
    }

    public class Client
    {
        public int ClientId { get; set; }
        public Bar Bar { get; set; }
        public int BarId { get; set; }
        public int? ClientNumber { get; set; }
    }

2) 将 ClientId 设置为标识(自动递增),将 ClientNumber 设置为生成的数据库(否​​则 EF Core 插入后不会从数据库中重新读取值)

entity.Property(e => e.ClientId).ValueGeneratedOnAdd();
entity.Property(e => e.ClientNumber).ValueGeneratedOnAddOrUpdate();

3) 添加触发器,修改插入后的ClientNumber

    create trigger ClientNumber on Clients after insert 
    as
    begin
        set nocount on;
        update c set ClientNumber = n.RowNum from Clients c
        inner join (
            select ClientId, ROW_NUMBER() OVER(PARTITION BY BarId ORDER BY ClientId ASC) as RowNum
            from Clients
        ) n on n.ClientId = c.ClientId
    where c.ClientId in (select ClientId from inserted)
    end

现在一切都在插入时自动工作:

            var bar1 = new Bar() { Clients = new List<Client>() };
            var bar2 = new Bar() { Clients = new List<Client>() };

            bar1.Clients.Add(new Client());
            bar1.Clients.Add(new Client());

            bar2.Clients.Add(new Client());
            bar2.Clients.Add(new Client());

            context.Bars.AddRange(new[] { bar1, bar2 });
            await context.SaveChangesAsync();

            bar1.Clients.Add(new Client());
            bar2.Clients.Add(new Client());
            bar1.Clients.Add(new Client());
            await context.SaveChangesAsync();

将渲染

ClientId    BarId   ClientNumber
1   1   1
2   1   2
6   1   3
7   1   4
3   2   1
4   2   2
5   2   3

缺点是您不能使用唯一索引 IX_CLIENT_NUMBER,因为在触发器执行之前,ClientNumber 将为 NULL。


推荐阅读