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();