EF Core如何处理多对多关系

简介: EF Core如何处理多对多关系

目录

一、解决多对多

二、增

三、查

四、删

EF Core在处理多对多关系时并不像一对一和一对多关系那样好处理,下面我们利用一个简单的电子商城购物车来讲解一下吧。


一、解决多对多

需求是这样的:用户可以将多个商品放入购物车,每个商品又属于多个购物车。我们先创建ShoppingCart和Commodity实体类。

public class ShoppingCart
{
    public int Id { get; set; }
    public ICollection<Commodity> Commoditys{ get; set; }
}
public class Commodity
{
    public int Id { get; set; }
    public string Name { get; set; }
    public int Quantity { get; set; }
    public ICollection<ShoppingCart> ShoppingCarts{ get; set; }
}

你第一眼看到这段代码是不是觉得这么做非常好?但是我要告诉你的是,到目前为止EF Core无法处理这样的代码,当你尝试添加迁移时控制台会输出如下内容:

Unable to determine the relationship represented by navigation property 'ShoppingCart.Commoditys' of type 'ICollection'. Either manually configure the relationship, or ignore this property using the '[NotMapped]' attribute or by using 'EntityTypeBuilder.Ignore' in 'OnModelCreating'.

那么我们该怎么做呢?聪明的同学一定想到了我们可以手动创建另一个中间表,它将建立ShoppingCart和Commodity多对多的关系。

public class ShoppingCartCommodity
{
    public int ShoppingCartId { get; set; }
    public ShoppingCart ShoppingCart{ get; set; }
    public int CommodityId { get; set; }
    public Commodity Commodity{ get; set; }
}

创建完中间表ShoppingCartCommodity,我们还要修改ShoppingCart和Commodity的导航属性:

public class ShoppingCart
{
    public int Id { get; set; }
    public ICollection<ShoppingCartCommodity> Commoditys { get; set; }
}
public class Commodity
{
    public int Id { get; set; }
    public string Name { get; set; }
    public int Quantity { get; set; }
    public ICollection<ShoppingCartCommodity> ShoppingCarts{ get; set; }
}

你以为这样处理完就完美了吗?NO!!!当你再次尝试添加迁移时会出现另一个错误提示:

The entity type 'ShoppingCart' requires a primary key to be defined.

ShoppingCart没有主键,由于多对多关系因此ShoppingCart应该是复合主键。复合主键由两列组成一个主键,在EF Core中创建复合键唯一办法是在OnModelCreating中创建。

protected override void OnModelCreating(ModelBuilder builder)
{
    base.OnModelCreating(builder);
    builder.Entity<ShoppingCartCommodity>().HasKey(p => new { p.ShoppingCartId, p.CommodityId});
}

到这里可以说才解决了EF Core处理多对多的问题。解决了多对多创建表的问题,下面我们就来看一下如何进行增删查。

二、增

我们要把商品添加到购物车中,我们需要创建ShoppingCartCommodity并保存它。

var shoppingCart= db.ShoppingCarts.First(i => i.Id == 1);
var commodity= db.Commoditys.First(i => i.Id == 2);
// 方法1:使用两个类的主键ID关联
var shoppingCartCommodity1= new ShoppingCartCommodity
{
    ShoppingCartId = shoppingCart.Id,
    CommodityId = commodity.Id
};
// 放法2:使用两个类实体关联
var shoppingCartCommodity2= new ShoppingCartCommodity
{
    ShoppingCart= cart,
    Commodity= item
};
db.Add(shoppingCartCommodity2);
db.SaveChanges();

三、查

从数据库中获取数据只需使用Include查询即可。

var shoppingCartIncludingCommoditys = db.Carts.Include(shoppingCart=> shoppingCart.Commoditys).ThenInclude(row => row.Commodity).First(shoppingCart=> shoppingCart.Id == 1);
// 获取指定购物车的所有商品
var shoppingCartCommodity2= shoppingCartIncludingCommoditys.Commoditys.Select(row => row.Commodity);
// 如果有购物车ID,则可以使用Linq获取所有商品:
var shoppingCartId = 1;
var shoppingCartCommoditys= db.Commoditys.Where(commodity=> commodity.shoppingCart.Any(j => j.ShoppingCartId== shoppingCartId));

四、删

如果要删除购物车中的商品时,可以这么做:

var shoppingCartId = 1;
var commodityId= 1;
var shoppingCartCommodity= db.ShoppingCartCommoditys.First(row => row.ShoppingCartId == shoppingCartId && row.CommodityId== commodityId);
db.Remove(shoppingCartCommodity);
db.SaveChanges();

如果要从购物车中删除所有项目,可以这么做:

var shoppingCart= db.ShoppingCarts.Include(c=> c.Commodity).First(i => i.Id == 2);
db.RemoveRange(shoppingCart.Commoditys);
db.SaveChanges();
目录
相关文章
|
Java
Java 对象间关系(依赖、关联、聚合和组合)
面向对象设计 对象间关系:依赖、关联、聚合和组合,四种关系容易混淆。特别后三种,只是在语义上有所区别,所谓语义就是指上下文环境、特定情景等。 
488 1
|
4月前
|
Java
neo4j仅仅删除关系如何操作
neo4j仅仅删除关系如何操作
256 5
|
3月前
|
开发框架 .NET 数据库连接
EF Core 在实际开发中,如何分层?
EF Core 在实际开发中,如何分层?
|
5月前
|
SQL 数据库
【Entity Framework】如何理解EF中的级联删除
【Entity Framework】如何理解EF中的级联删除
54 0
|
Java
java 一对多、多对多关系示例
java 一对多、多对多关系示例
327 0
|
前端开发 JavaScript
【组件封装】基于neo4jD3封装关系图、关联图谱
【组件封装】基于neo4jD3封装关系图、关联图谱
【组件封装】基于neo4jD3封装关系图、关联图谱
|
数据库
一对多关系domain Model中设置使用AutoMapper时出错
一对多关系domain Model中设置使用AutoMapper时出错
171 0
一对多关系domain Model中设置使用AutoMapper时出错