实体框架代码优先-每个类型的表继承-插入问题?

如何解决实体框架代码优先-每个类型的表继承-插入问题?

| 我在插入继承自基类的子类的实例时遇到问题。 考虑以下来自这些POCO的代码段:
public abstract class EntityBase<T> 
{   
    private T _id;       
    [Key]
    public T ID    
    {       
         // get and set details ommitted.   
    }
}
public abstract class PersonBase<T> : EntityBase<T>
{   
    // Details ommited.  
}

public class Applicant : PersonBase<int>
{   
    // Details ommitted for brevity.  
}

public class Employee : Applicant {}   
现在相当标准的继承。在我们的系统中,当申请人最终成为雇员时,我们会收集额外的数据。如果不雇用他们,他们仍然是申请人,信息有限。 现在考虑建立每个类型表继承模型的流畅API:
protected override void OnModelCreating(DbModelBuilder modelBuilder)
{
     base.OnModelCreating(modelBuilder);                
     // Set up table per type object mapping for the Visitor Hierarchy.       
     modelBuilder.Entity<Employee>().ToTable(\"Employees\");
} 
到目前为止,一切都很好... 如果我查看由此创建的数据库,则有一个名为Applicants的表,其ID列的类型为int,自动递增ID,并且有一个Employees表,其ID字段为主键(非自动递增)。 基本上,雇员表中的ID字段是申请人表的外键。 这就是我要的。我不希望在与申请人表相对应的雇员表中进行记录,直到他们实际成为雇员为止。 当我尝试插入一个属于以下代码的Employee时,问题就来了:
public void PersistCreationOf(T entity)
{     
    DataContextFactory.GetDataContext().Set(typeof(T)).Add(entity);
} 
问题:它插入了一个全新的申请人和雇员。我将其连接到Sql Profiler,并查看了两个插入查询。 我只想插入雇员记录及其已经拥有的ID(Visitors表中的外键)。 我了解默认情况下需要这样做:显然,如果创建一个子类并将其插入,则需要将其插入两个表中。 我的问题是可以告诉框架-基本表已经有信息-只是插入子表中? 提前致谢!     

解决方法

        除了发送原始SQL命令以将“ 3”减“ 4”属性片段插入“ 5”表之外,我相信这是不可能的。您可以更新或插入实体。基本上,您想要的是更新
Employee
的基本部分(如果什么都没有更改,则什么也不做)并插入不可能的派生部分。 想象一下ORM的作用:它将数据库中的关键身份映射到内存中的对象身份。即使在内存中,您也无法实现所需的目标:如果内存中有一个对象为
Applicant
,则它始终是申请人。您无法神奇地将其“升级”到3英镑。您将必须创建一个类型为“ 3”的新对象,将“ 4”的属性复制到新的“ 3”的基本属性中,然后删除该“ 4”。结果是具有新对象标识的新对象。 我认为您必须在EF中遵循相同的步骤。您的“ 3”将是一个新实体,在“ 4”和“ 3”表中都有新行,您需要删除旧的“ 4”。如果您具有自动生成的密钥,它将是具有新ID的新身份。 (如果您没有自动生成的密钥,则可以在删除旧的
Applicant
之后再次提供旧的ID,从而“伪造”不变的身份。)如果您引用具有以下内容的旧申请人,这当然会带来很大的潜在麻烦FK约束。 在这种情况下,将申请者“升级”为员工可能不是最佳的继承方法。 “ 4”内部的可选导航属性(从1到0 ... 1的关系)是指另一个包含其他属性的实体,这些属性使申请人成为雇员可以解决此问题。可以设置或不设置此导航属性,从而使您可以区分申请人和也是雇员的申请人。并且,当您将其设为员工时,您无需删除和更改申请人的ID。 (正如所说的“我相信”。也许有一种隐藏的方式,我没有看到。)     

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐


依赖报错 idea导入项目后依赖报错,解决方案:https://blog.csdn.net/weixin_42420249/article/details/81191861 依赖版本报错:更换其他版本 无法下载依赖可参考:https://blog.csdn.net/weixin_42628809/a
错误1:代码生成器依赖和mybatis依赖冲突 启动项目时报错如下 2021-12-03 13:33:33.927 ERROR 7228 [ main] o.s.b.d.LoggingFailureAnalysisReporter : *************************** APPL
错误1:gradle项目控制台输出为乱码 # 解决方案:https://blog.csdn.net/weixin_43501566/article/details/112482302 # 在gradle-wrapper.properties 添加以下内容 org.gradle.jvmargs=-Df
错误还原:在查询的过程中,传入的workType为0时,该条件不起作用 &lt;select id=&quot;xxx&quot;&gt; SELECT di.id, di.name, di.work_type, di.updated... &lt;where&gt; &lt;if test=&qu
报错如下,gcc版本太低 ^ server.c:5346:31: 错误:‘struct redisServer’没有名为‘server_cpulist’的成员 redisSetCpuAffinity(server.server_cpulist); ^ server.c: 在函数‘hasActiveC
解决方案1 1、改项目中.idea/workspace.xml配置文件,增加dynamic.classpath参数 2、搜索PropertiesComponent,添加如下 &lt;property name=&quot;dynamic.classpath&quot; value=&quot;tru
删除根组件app.vue中的默认代码后报错:Module Error (from ./node_modules/eslint-loader/index.js): 解决方案:关闭ESlint代码检测,在项目根目录创建vue.config.js,在文件中添加 module.exports = { lin
查看spark默认的python版本 [root@master day27]# pyspark /home/software/spark-2.3.4-bin-hadoop2.7/conf/spark-env.sh: line 2: /usr/local/hadoop/bin/hadoop: No s
使用本地python环境可以成功执行 import pandas as pd import matplotlib.pyplot as plt # 设置字体 plt.rcParams[&#39;font.sans-serif&#39;] = [&#39;SimHei&#39;] # 能正确显示负号 p
错误1:Request method ‘DELETE‘ not supported 错误还原:controller层有一个接口,访问该接口时报错:Request method ‘DELETE‘ not supported 错误原因:没有接收到前端传入的参数,修改为如下 参考 错误2:cannot r
错误1:启动docker镜像时报错:Error response from daemon: driver failed programming external connectivity on endpoint quirky_allen 解决方法:重启docker -&gt; systemctl r
错误1:private field ‘xxx‘ is never assigned 按Altʾnter快捷键,选择第2项 参考:https://blog.csdn.net/shi_hong_fei_hei/article/details/88814070 错误2:启动时报错,不能找到主启动类 #
报错如下,通过源不能下载,最后警告pip需升级版本 Requirement already satisfied: pip in c:\users\ychen\appdata\local\programs\python\python310\lib\site-packages (22.0.4) Coll
错误1:maven打包报错 错误还原:使用maven打包项目时报错如下 [ERROR] Failed to execute goal org.apache.maven.plugins:maven-resources-plugin:3.2.0:resources (default-resources)
错误1:服务调用时报错 服务消费者模块assess通过openFeign调用服务提供者模块hires 如下为服务提供者模块hires的控制层接口 @RestController @RequestMapping(&quot;/hires&quot;) public class FeignControl
错误1:运行项目后报如下错误 解决方案 报错2:Failed to execute goal org.apache.maven.plugins:maven-compiler-plugin:3.8.1:compile (default-compile) on project sb 解决方案:在pom.
参考 错误原因 过滤器或拦截器在生效时,redisTemplate还没有注入 解决方案:在注入容器时就生效 @Component //项目运行时就注入Spring容器 public class RedisBean { @Resource private RedisTemplate&lt;String
使用vite构建项目报错 C:\Users\ychen\work&gt;npm init @vitejs/app @vitejs/create-app is deprecated, use npm init vite instead C:\Users\ychen\AppData\Local\npm-