EF基础知识小记七(拆分实体到多个表以及拆分表到多个实体)

本文介绍了在软件开发过程中,如何通过拆分实体到多个表或拆分一张表到多个实体的方法来优化系统架构,提高系统性能。

一、拆分实体到多个表

1、在日常开发中,会经常碰到一些老系统,当客户提出一些新的需求,这些需求需要在原来的表的基础上加一些字段,大多数人会选择通过给原表添加字段的方式来完成这些需求,方法,虽然可行,但是如果架构不合理的系统,就会牵一发而动全身.所以处理这种需求比较合理的方式是:建一张新表来存放新的字段.

通过叫做合并两张及以上的表到一个单独的实体,也叫分拆一个实体到多个表,我们把每个组成部分当成一个逻辑实体.这个过程叫做逻辑分拆.

缺点:每当获取实体时,框架都需要额外的Join联结.

 

2、示例

下面通过一个示例简单介绍下逻辑分拆

(1)、数据库表设计图

(2)、编写代码

i、确认目标项目导入了EF的相关程序集

ii、创建Product实体,代码如下:

    public class Product
    {
        [Key]
        [DatabaseGenerated(DatabaseGeneratedOption.Identity)]
        public int SKU { get; set; }

        public string Description { get; set; }
        public decimal Price { get; set; }
        public string ImgUrl { get; set; }
    }

iii、创建数据上下文对象,该对象必须继承DbContext,代码如下:

public class EF6RecipesContext:DbContext
    {
        public DbSet<Product> Products { get; set; }

        public EF6RecipesContext():base("name=EF6RecipeEntities")
        {
        }

        protected override void OnModelCreating(DbModelBuilder modelBuilder)
        {
            base.OnModelCreating(modelBuilder);
            modelBuilder.Entity<Product>()
                .Map(m => {
                    m.Properties(p => new { p.Sku, p.Description, p.Price });
                    m.ToTable("Product", "dbo");
                })
                .Map(m => {
                    m.Properties(p => new { p.Sku, p.ImgUrl });
                    m.ToTable("ProductWebInfo", "dbo");
                });
        }
    }

iiii、编写测试代码:

 

            using (var context = new EF6RecipesContext())
            {
                var product = new Product
                {
                    Sku = 1,
                    Description = "啦啦啦",
                    Price = 1M,
                    ImgUrl = "1.jpg"
                };
                context.Products.Add(product);
                product = new Product
                {
                    Sku = 2,
                    Description = "哈哈哈",
                    Price = 2M,
                    ImgUrl = "2.jpg"
                };
                context.Products.Add(product);
                product = new Product
                {
                    Sku = 3,
                    Description = "呵呵呵",
                    Price = 3M,
                    ImgUrl = "3.jp"
                };
                context.Products.Add(product);
                context.SaveChanges();
            }
            using (var context = new EF6RecipesContext())
            {
                foreach (var p in context.Products)
                {
                    Console.WriteLine("{0} {1} {2} {3}", p.Sku, p.Description,p.Price.ToString("C"), p.ImgUrl);
                }
            }
            Console.ReadKey();

 

 

 二、拆分一张表到多个实体

假设数据库中有一张表,里面包含一些常用的字段,但是也包含一些不常用的大字段。为了提供系统的性能,需要避免每个查询都去加载这些字段.这个时候我们就需要将表拆分成两个或者更多的实体.

 

C# 中,EF(Entity Framework) 可以通过使用自定义实体类来将多个实体类的状态保存在一张状态中。以下是一些基本步骤: 1. 创建一个新的实体类,用于示状态。该类应该包含用于识别每个实体类和保存它们的状态的属性。 2. 在每个要保存状态的实体类中添加一个引用到状态实体类的属性。 3. 使用 EF 的 Code First 功能将实体类映射到数据库中。在这里,你需要指定状态实体类和每个实体类的引用属性的映射方式。 4. 在需要保存实体类状态的时候,实例化状态实体类,并将每个实体类的状态设置到它们对应的引用属性中。然后将状态实体类添加到 EF 上下文中,并调用 SaveChanges 方法来将它们保存到数据库中。 下面是一个示例代码,演示了如何使用 C# EF多个实体类的状态保存在一张状态中: ```csharp public class Status { public int Id { get; set; } public string EntityName { get; set; } public string EntityState { get; set; } } public class Entity1 { public int Id { get; set; } public string Name { get; set; } public virtual Status Status { get; set; } } public class Entity2 { public int Id { get; set; } public int Age { get; set; } public virtual Status Status { get; set; } } public class MyContext : DbContext { public DbSet<Entity1> Entity1Set { get; set; } public DbSet<Entity2> Entity2Set { get; set; } public DbSet<Status> StatusSet { get; set; } protected override void OnModelCreating(DbModelBuilder modelBuilder) { modelBuilder.Entity<Entity1>() .HasOptional(e => e.Status) .WithMany() .HasForeignKey(e => e.StatusId); modelBuilder.Entity<Entity2>() .HasOptional(e => e.Status) .WithMany() .HasForeignKey(e => e.StatusId); } } // 保存状态 using (var context = new MyContext()) { var entity1 = context.Entity1Set.Find(1); var entity2 = context.Entity2Set.Find(1); var status = new Status { EntityName = "Entity1", EntityState = "Modified", }; entity1.Status = status; status = new Status { EntityName = "Entity2", EntityState = "Modified", }; entity2.Status = status; context.StatusSet.Add(status); context.SaveChanges(); } ``` 在这个例子中,我们定义了三个实体类:Entity1、Entity2 和 Status。Entity1 和 Entity2 都有一个指向 Status 实体类的引用属性。我们使用 EF 的 Fluent API,将它们与 Status 实体类映射到数据库中,并指定它们的外键属性。 最后,我们实例化了一个状态实体类,将 Entity1 和 Entity2 的状态设置到它们对应的引用属性中,并将状态实体类添加到 EF 上下文中。最后,我们调用 SaveChanges 方法,将它们保存到数据库中。 这是一个简单的示例,你可以根据你的需求进行修改和扩展。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值