EF Codefirst 多对多关系 操作中间表的 增删改查(CRUD)
前言
此文章只是为了给新手程序员,和经验不多的程序员,在学习ef和lambada表达式的过程中可能遇到的问题。
本次使用订单表和员工表建立多对多关系。
首先是订单表:
public class Order
{
public int OrderId { get; set; } public string OrderTitle { get; set; } public string CustomerName { get; set; } public DateTime TransactionDate { get; set; } [ConcurrencyCheck]
[Timestamp]
public byte[] TimeStamp { get; set; } public virtual ICollection<Employee> InvolvedEmployees { get; set; }
}
接下来是员工表:
public class Employee
{
public int EmployeeId { get; set; } public string EmployeeName { get; set; } public virtual ICollection<Order> Orders { get; set; }
}
映射文件(mapping):
public class OrderMap:EntityTypeConfiguration<Order>
{
public OrderMap()
{
this.HasKey(o => o.OrderId);
//OrderId为自增长
this.Property(o => o.OrderId).HasDatabaseGeneratedOption(DatabaseGeneratedOption.Identity);
this.Property(o => o.OrderTitle).IsRequired().HasMaxLength();//订单名称为必填,最大长度为64;
this.Property(o => o.CustomerName).IsRequired().HasMaxLength();//订单名称为必填,最大长度为64;
this.Property(o => o.TransactionDate).IsRequired(); //订单名称为必填,最大长度为64;
}
}
public class EmployeeMap:EntityTypeConfiguration<Employee>
{
/// <summary>
/// 构造函数
/// </summary>
public EmployeeMap()
{
this.HasKey(x => x.EmployeeId);
this.ToTable("Employees");
this.Property(x => x.EmployeeName).IsRequired().HasMaxLength(); //设置多对多的关系 .Map()配置用于存储关系的外键列和表。
/*
Employees HasMany此实体类型配置一对多关系。对应Orders实体
WithMany 将关系配置为 many:many,且在关系的另一端有导航属性。
* MapLeftKey 配置左外键的列名。左外键指向在 HasMany 调用中指定的导航属性的父实体。
* MapRightKey 配置右外键的列名。右外键指向在 WithMany 调用中指定的导航属性的父实体。
*/
this.HasMany(x => x.Orders).
WithMany(x => x.InvolvedEmployees).
Map(m => m.ToTable("EmployeeOrder").
MapLeftKey("EmployeeId").
MapRightKey("OrderId"));
}
}
dbcontext文件:
public class EfCodeFirstWithManyDbContext:DbContext
{
public EfCodeFirstWithManyDbContext()
: base("DefaultConnection")
{
}
public IDbSet<Order> Orderses { get; set; }
public IDbSet<Employee> Employeees { get; set; }
protected override void OnModelCreating(DbModelBuilder modelBuilder)
{
modelBuilder.Configurations.Add(new OrderMap());
modelBuilder.Configurations.Add(new EmployeeMap());
base.OnModelCreating(modelBuilder);
}
}
生成数据库:

数据库关系图:

基础工作建立完毕。
正文:
我们都知道在codefirst 配置多对多关系的时候,会自动给我买生成中间表。
在modelfirst和datafirst的时候都会生成一个中间类:EmployeeOrder.cs
而codefirst不会生成这个类,本文所阐述的就是使用在codefirst中使用ef、lambada表达对其进行增删改查的多种情况
创建一个订单
添加订单信息、员工信息到数据表中,建立两则多对多的联系
清空中间表之间的数据,而不影响employee和order表中的数据
给中间表添加数据,给两个已经存在的数据建立中间关系
操作中间表,修改两个表employee和order中值,并且删除中间表中多余的值
本文大概操作次5种情况。
代码:
//添加订单信息、员工信息到数据表中,建立两则多对多的联系
public static void CreateFullOrderByEmployee()
{
#region 添加订单信息、员工信息到数据表中,建立两则多对多的联系 using (var dbContext = new EfCodeFirstWithManyDbContext())
{
var order = new Order
{
OrderTitle = "购买汽车",
CustomerName = "梁桐铭",
TransactionDate = DateTime.Now,
InvolvedEmployees = new List<Employee>()
};
var employee1 = new Employee {EmployeeName = "管理员-yoyocms", Orders = new List<Order>()};
var employee2 = new Employee {EmployeeName = "主管-yoyocms", Orders = new List<Order>()};
//先保存订单到数据库中
dbContext.Orderses.Add(order); order.InvolvedEmployees.Add(employee1);
//order.InvolvedEmployees.Add(employee2); // employee2.Orders.Add(order);
var res = dbContext.SaveChanges();
} #endregion
}
为了测试方便对这个方法for循环了20次:
private static void Main(string[] args)
{ for (int i = ; i < ; i++)
{
CreateFullOrderByEmployee();
} Console.WriteLine("加载完毕,请点击任意键退出");
Console.ReadKey();
}
清空中间表信息,而不影响order表和employee表的信息
//清空两个中间表之间的关系
public static void EmptyEmployeeOrder()
{
using (var dbContext = new EfCodeFirstWithManyDbContext())
{
//获取到employeeId为20下,所有Orders订单列表信息和员工信息。
var employeeToUpdate =
dbContext.Employeees.Include(x => x.Orders).FirstOrDefault(x => x.EmployeeId == );
if (employeeToUpdate != null)
{
employeeToUpdate.Orders = new List<Order>();
dbContext.SaveChanges();
}
else
{
Console.WriteLine("查询失败EmptyEmployeeOrder为空");
}
}
}
建立员工表和对应的订单表中建立两个表之间的联系
//建立两个已经存在的数据建立中间关系
public static void AddInfoEmployeeOrder()
{
using (var dbContext = new EfCodeFirstWithManyDbContext())
{
//获取到employeeId为20下,所有Orders订单列表信息和员工信息。
var employeeToAdd = dbContext.Employeees.Include(x => x.Orders).FirstOrDefault(x => x.EmployeeId == );
//设计订单表的集合,将新增的数据填充进来
int[] orderIdList = {, , , , , , };
//判断employeeToAdd.Orders中是否有重复的OrderId
if (employeeToAdd != null)
{
//查询出目前员工对应的订单表
var employeeOrder = new HashSet<int>(employeeToAdd.Orders.Select(e => e.OrderId)); foreach (var order in dbContext.Orderses)
{
//即将要添加orderIdList值的是否包含订单表的id
//筛选出orderidList和orders中共同的值,添加到order.OrderId
if (orderIdList.Contains(order.OrderId))
{
//查询出目前employee表中的orderId是否包含了orderIdList中的id
if (employeeOrder.Contains(order.OrderId))
{
//打印出重复的orderId
Console.WriteLine("重复的ID为" + order.OrderId);
Console.WriteLine("不执行添加结果");
}
else
{
//打印出Employee表中没有orderId
Console.WriteLine("即将添加的值" + order.OrderId);
//添加重复的值
employeeToAdd.Orders.Add(order);
}
}
}
}
else
{
Console.WriteLine("employeeToAdd信息为空");
} dbContext.SaveChanges();
}
}
修改两个表employee和order中值,并且删除多余的值
/// <summary>
/// 修改两个表employee和order中值,并且删除多余的值
/// </summary>
public static void UpdateInfoEmployeeOrder()
{
//首先获取到EmployeeId=20中,所有的Orders列表和employee信息
using (var dbContext = new EfCodeFirstWithManyDbContext())
{
var employeeUpdate =
dbContext.Employeees.Include(x => x.Orders).FirstOrDefault(x => x.EmployeeId == );
//需要更改对应的OrderId列表
int[] orderIdList = {, , , , , , }; if (employeeUpdate != null)
{
//获取employee中的OrderIdlist
var employeeOrderIdList = new HashSet<int>(employeeUpdate.Orders.Select(e => e.OrderId));
foreach (var order in dbContext.Orderses)
{
//判断要修改的Orderid和Orders表中的均包含
if (orderIdList.Contains(order.OrderId))
{
if (!employeeOrderIdList.Contains(order.OrderId))
{
Console.WriteLine("修改对应的订单Id表" + order.OrderId);
employeeUpdate.Orders.Add(order);
}
}
else
{
if (employeeOrderIdList.Contains(order.OrderId))
{
Console.WriteLine("删除无用的订单表id"+order.OrderId);
employeeUpdate.Orders.Remove(order);
}
}
}
}
else
{
Console.WriteLine("查无employeeUpdate 的信息");
} dbContext.SaveChanges();
}
}
尾声
至此操作实现了对codefirst中,对中间表的CRUD过程。
EF Codefirst 多对多关系 操作中间表的 增删改查(CRUD)的更多相关文章
- 国产化之路-统信UOS + Nginx + Asp.Net MVC + EF Core 3.1 + 达梦DM8实现简单增删改查操作
专题目录 国产化之路-统信UOS操作系统安装 国产化之路-国产操作系统安装.net core 3.1 sdk 国产化之路-安装WEB服务器 国产化之路-安装达梦DM8数据库 国产化之路-统信UOS + ...
- MyBatis-单表的增删改查(CRUD)操作
在学习MyBatis的单表的增删改查操作之前,还是再次熟悉下MyBatis这个框架,只有对其熟悉的情况下,才能很好的使用,灵活的开发. MyBatis优点: ...
- Hibernate5笔记2--单表的增删改查操作
单表的增删改查操作: (1)定义获取Session和SessionFactory的工具类: package com.tongji.utils; import org.hibernate.Session ...
- python全栈开发day61-django简单的出版社网站展示,添加,删除,编辑(单表的增删改查)
day61 django内容回顾: 1. 下载: pip install django==1.11.14 pip install -i 源 django==1.11.14 pycharm 2. 创建项 ...
- django模型层 关于单表的增删改查
关于ORM MTV或者MVC框架中包括一个重要的部分,就是ORM,它实现了数据模型与数据库的解耦,即数据模型的设计不需要依赖于特定的数据库, 通过简单的配置就可以轻松更换数据库,这极大的减轻了开发人员 ...
- Mysql数据表的增删改查
---恢复内容开始--- Mysql数据表的增删改查 1.创建表 语法:CREATE TABLE 表名(字段1,字段2,字段3.......) CREATE TABLE `users` ( `us ...
- ORM 实现数据库表的增删改查
这次通过反射技术来实现一下数据库表的增删改查对象关系映射(英语:Object Relational Mapping,简称ORM,或O/RM,或O/R mapping) 注:引用时约束了以下几点: 数据 ...
- day 57 data 插件 表的增删改查
一 data的含义 在匹配的元素集合中的所有元素上存储任意相关数据或返回匹配的元素集合中的第一个元素的给定名称的数据存储的值. 1 .data(key, value): 描述:在匹配 ...
- springLdap 操作ldap示例(增删改查)
转自:http://blog.csdn.net/sundenskyqq/article/details/9002440 这部分的示例网上的确有很多,但是个人在查找的过程中还是感到不够满意,所以就自己总 ...
随机推荐
- Drools 函数学习
Drools 函数学习 函数是定义在规则文件当中一代码块,作用是将在规则文件当中若干个规则都会用到的业务操作封装起来,实现业务代码的复用,减少规则编写的工作量.函数的编写位置可以是规则文件当中 pac ...
- 安装zeppelin
安装zeppelin 1.默认安装好spark集群 2.安装zeppelin 1.解压安装包 tar zxvf zeppelin-0.5.5-incubating-bin-all.tgz 2.配置环境 ...
- SQL 语句与性能之联合查询和联合分类查询
select * from t1 left join t2 on t2.sysno =t1.ASysNo left join t3 on t3.sysno =t2.ASysNo left join t ...
- Git安装
前面说了很多废话,说得再好,还不如实践一次.要想使用Git,首先得安装.这次实验主要是实践怎样安装Git. 安装与检查是否安装成功 1. 在Linux(Ubuntu)上安装 上篇的Git简介已经介绍过 ...
- maven执行报错resolution will not be reattempted until the update interval of nexus h
maven在执行过程中抛错: 引用 ... was cached in the local repository, resolution will not be reattempted until t ...
- twitter.common.concurrent deadline and defer
此defer非golang中的defer https://tour.golang.org/flowcontrol/12 from twitter.common.concurrent import Ti ...
- Arduino下LCD1602综合探究(下)——如何减少1602的连线,LiquidCrystal库,LiquidCrystal库中bug的解决方法
一.前言: 上文中,笔者系统的阐述了1602的两种驱动方式,并简单的提到了Arduino的LiquidCrystal库.本文紧接上文,对以下两个问题进行更加深入的探讨:如何能够使1602对Arduin ...
- appzapper注册码
Appzapper for mac是MAC OS上的一款软件,可以非常方便的卸载自己不喜欢的软件,非常的快速便捷,卸载的时候不会有残留. 下载地址:http://www.pc6.com/mac/114 ...
- BZOJ2498 : Xavier is Learning to Count
考虑容斥,通过$Bell(p)$的时间枚举所有等价情况. 对于一种情况,强制了一个等价类里面的数都要相同,其它的可以相同也可以不同. 这方案数显然可以通过多项式乘法求得,乘上容斥系数$(-1)^{p- ...
- android 从 phonegap 到 js webview 交互
像生活类.办公协同类. 动态添加,下载等. 1.phonegap 我这里用了旧的版本,可能新版本变化大了. 创建asset资源文件夹,然后新建index.html copy 相应的js 文件进来. 创 ...