c# - 自动递增数字的最佳实践 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 实体.
解决方案
一种方法是执行以下操作;
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。
推荐阅读
- spring - 当我构建弹簧源时,GroovyDynamicElementReader 无法解析为类型
- installation - 使用 npm install -g appium 通过节点的 Appium 安装问题
- python - Github Actions 停留在一个任务上,最后超时失败
- oracle - 由于 out 参数分配错误,dbms_scheduler 作业失败
- c++ - std::_String_alloc 析构函数链接器错误
- javascript - 是否可以从数据集中使 D3 图表图例颜色动态化?
- python - 如何使用随机数值更新模板
- ios - 如何在导航栏下添加一个可以在所有视图控制器中使用和显示的视图?
- spring-integration - Spring Integration Sftp 上传失败,但简单的 Jsch 上传是可以的。为什么?
- angular - 如何使用选择下拉菜单过滤数据表