- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
我有一个与 Entity Framework 映射的 SQLite 数据库。有 2 个表:Collections (1:n) Albums。
当我删除一个集合时,所有相关的相册也必须被删除。我使用 CollectionRepo.Delete(collection);
来实现这一点。它使用以下代码:
public int Delete(Collection entity)
{
Context.Entry(entity).State = EntityState.Deleted;
return Context.SaveChanges();
}
问题是:当我执行这段代码时,Context.SaveChanges();
给我一个异常:
The operation failed: The relationship could not be changed because one or more of the foreign-key properties is non-nullable. When a change is made to a relationship, the related foreign-key property is set to a null value. If the foreign-key does not support null values, a new relationship must be defined, the foreign-key property must be assigned another non-null value, or the unrelated object must be deleted.
似乎 Entity Framework 想要在外键上 null
而不是删除条目。但这绝对不是我想要的,因为没有 parent 的相册毫无意义(至少在我的用例中)。
我显然可以先手动删除相册,然后再删除空的集合,但在我看来这有点棘手。首先,在我看来,EF 应该足够聪明,可以自己完成它以简化代码;其次,如果我有几十个与收藏和相册的关系,我最终会得到一个相当大、难以维护的关系,代码库。
集合类
public class Collection
{
[Key, DatabaseGenerated(DatabaseGeneratedOption.Identity)]
public long Id { get; set; }
public virtual List<Album> Albums { get; set; } = new List<Album>();
}
专辑类
public class Album
{
[Key, DatabaseGenerated(DatabaseGeneratedOption.Identity)]
public long Id { get; set; }
[Required]
[ForeignKey("Collection")]
public long CollectionId { get; set; }
public virtual Collection Collection { get; set; }
}
DbContext 子类
public class DataEntities : DbContext
{
public virtual DbSet<Collection> Collections { get; set; }
public virtual DbSet<Album> Albums { get; set; }
public DataEntities() : base("name=Connection")
{
Configuration.ProxyCreationEnabled = false;
}
protected override void OnModelCreating(DbModelBuilder modelBuilder)
{
modelBuilder.Entity<Album>()
.HasRequired(a => a.Collection)
.WithMany(c => c.Albums)
.HasForeignKey(a => a.CollectionId)
.WillCascadeOnDelete(true);
modelBuilder.Entity<Collection>()
.HasMany(c => c.Albums)
.WithRequired(a => a.Collection)
.WillCascadeOnDelete(true);
}
}
最佳答案
应用分离对象图修改在 EF 中一直不清楚。这是无故失败的情况之一。
假设 Collection
传递给 Delete
的实体方法有Albums
集合填充(至少这是我能够重现异常的方式)。线路
Context.Entry(entity).State = EntityState.Deleted;
做两件事:附加 entity
和所有 Album
来自 entity.Albums
的对象根据上下文,标记 entity
作为Deleted
,和(注意!)Album
对象为 Modified
.当您调用 SaveChanges
时,这会导致不正确的行为,最后生成有问题的异常。
有两种方法(变通方法)可以修复这种不正确的行为。
第一个是将上面一行替换为
Context.Collections.Attach(entity);
Context.Collections.Remove(entity);
效果与上述类似,重要和不同之处在于现在相关的Album
标记为 Deleted
的对象, 这允许成功执行 SaveChanges
.
缺点是现在 SaveChanges
问题 DELETE
每个 Album
的命令在删除 Collection
的命令之前,这是低效的并且没有多大意义,因为级联删除将在数据库内完美地处理它。
第二个选项是保持代码不变,但在附加实体之前清除相关集合:
entity.Albums = null;
Context.Entry(entity).State = EntityState.Deleted;
这允许成功执行 SaveChanges
并生成一个 DELETE
仅针对实体的命令。
缺点是您需要编写额外的代码并且不要忘记任何支持级联删除的子集合,而不是为需要级联更新的集合执行此操作(即具有可选关系,需要使用 NULL
更新 FK 字段) .
选择权在你。
关于c# - Entity Framework 硬级联删除,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/41858097/
算力时代,视频云需要怎样的 CPU? 在数据爆发式增长及算法日益精进的大背景下,属于「算力」的时代俨然到来。随着视频成为互联网流量的主角,日趋饱和的音视频场景渗透率、人类对“感官之限”的追
我正在使用 keras 开发深度网络。有一个激活“硬 sigmoid”。它的数学定义是什么? 我知道什么是 Sigmoid。有人在Quora上问了类似的问题:https://www.quora.com
我有一个不寻常的 SQL 问题,我不太确定如何最好地解释,所以请耐心等待。我有三张表,一张是志愿者组织的表,一张是用户的表,一张是用户详细信息的表。 #Table 1# ## Name Preside
我正在尝试使用名为 bigText 的 jquery 插件。一个很棒的用于创建 block 头的插件。如果您想将其与自定义字体一起使用,它会声明您需要 google webfont loader,这样
假设我有一张 table date,personid 1/1/2001 1 1/2/2001 3 1/3/2001 2 1/4/2001 2 1/5/2001 5 1/6/2001 5 1/7/200
下面是我要执行的 SQL。我想避免为此执行多个请求,我很确定这是可能的…… First table : products_categories (category_id, category_infos
我在 android studio 中重新设置了一些提交,并选择了硬重置类型。我失去了一个星期的工作。是否有希望撤销此操作?我正在使用 android studio,它有内置的 GUI 选项来执行所有
当我使用我的交叉工具链编译 C 代码时,链接器会打印出警告页面,说明我的可执行文件使用了硬 float ,但我的 libc 使用了软 float 。有什么区别? 最佳答案 硬浮点使用片上浮点单元。软
linux系统有arm64,arm架构armv8-a。如何知道 debian 是运行硬浮点还是软浮点? 最佳答案 符合 AAPCS64, GNU GCC for armv8 仅提供硬浮点 aarch6
我正在开发 cortex-m3 的微内核。我创建了一个故意导致错误的小型测试应用程序。 现在我不确定如何从故障中返回。我知道堆栈可能需要使用不同函数的地址进行更新。我也知道在某些情况下从错误返回可能是
硬/软 限制是什么意思? 核心文件大小的差异例如: ulimit -Sc 1024 与 ulimit -Hc 1024 我通常在运行二进制文件之前将脚本放入 ulimit -c unlimited。
我想在 Java 中加载一个 MSCAPI keystore 并检查 MY 存储中的可用证书。但是,这些证书的一些 key 驻留在硬件 token 上,并且弹出窗口会在加载期间询问 token 。 有
是的,这是一个有点棘手的问题; 一个数组(没有副本),而不是任何奇数数组。让我解释一下,让我们从这里开始; $a = array ( 'one' => 1, 'two' => 2, 'three' =
我需要在运行 Ubuntu 12.04 的 BeagleBoard xM rev C 上运行一个使用 ftd2xx 的程序。我正在尝试使用提供的 ARM 库 libftd2xx.so here . l
我是一名优秀的程序员,十分优秀!