现在使用多数的数据库是关系型数据库,那么表与表之间的关系就会显得尤其重要,对于数据的CRUD处理和以后数据的分析有很大的好处。下面是对于数据库中对表关系的理解以及在EF中使用Fluent API来创建这种关系的例子.

数据库中实体之间的联系

书中语录:在现实世界中,事务内部以及事务之间是有联系的,这些联系在计算机里面叫做实体之间的联系和实体内部的联系,内部的联系就是组成实体属性之间的联系,而实体之间的联系可以分为三类。(1:1, 1:n, n:n)

一:一对一联系(1:1)

A实体集中每一个实体在实体集B中至多有一个实体与之联系,反之亦然,这就是一对一联系。

eg:一个班只有一个班长,而一个班长只在一个班中任职。班级与班长就是一对一关系。

属性模型

    /// <summary>
/// Monitor代表班长类
/// </summary>
public class Monitor
{
public Guid MonitorId { get; set; }
public string MName { get; set; }
public int MAge { get; set; }
public virtual TheClass TheClass { get; set; } //导航属性
}
/// <summary>
/// TheClass表示班级
/// </summary>
public class TheClass
{
public TheClass()
{
this.Students = new HashSet<Student>();
}
public virtual Monitor Monitor { get; set; } // 导航属性
public Guid TheClassId { get; set; }
public string TcName { get; set; }
public int TcNumber { get; set; }
public virtual ICollection<Student> Students { get; set; } //班级里面有很多学生
}

Fluent API(1:1)

    public class MonitorMap : EntityTypeConfiguration<Monitor>
{
public MonitorMap()
{
this.ToTable("Monitor");
this.HasKey(x=>x.MonitorId);
Property(x => x.MName).HasMaxLength(128);
Property(x => x.MAge);
this.HasRequired(x => x.TheClass).WithOptional(x => x.Monitor).Map(x => x.MapKey("TheClassId")); //这里设置Monitor为依赖类,且TheClassId为外键。
}
} public class TheClassMap:EntityTypeConfiguration<TheClass>
{
public TheClassMap()
{
ToTable("TheClass");
HasKey(x => x.TheClassId);
Property(x => x.TcName).HasMaxLength(128);
Property(x => x.TcNumber);
HasMany(x => x.Students);
HasOptional(x => x.Monitor);
}
}

数据库中的表

在实现一对一的过程中,出现了很多的问题,说是外键设置的问题,依赖类和主类关系不清.这些刚开始的时候一直出错,导致数据库中模型创建不了,浪费很长时间,出现这种情况就是我们代码里面设置的问题,此时我们百度,博客园等均可解决 ,上面Fluent API代码中是正确的。

二:一对多联系(1:n)

A实体集中每一个实体在B实体集中有n个实体与之对应(n>=0),反过来,B实体集中的每一个实体在A中只有1个与之对应。这种形式就是一对多联系。

eg:班集和班集里面学生的关系。学校和学生的关系。

属性

    /// <summary>
/// TheClass表示班级
/// </summary>
public class TheClass
{
public TheClass()
{
this.Students = new HashSet<Student>();
}
public virtual Monitor Monitor { get; set; } // 导航属性
public Guid TheClassId { get; set; }
public string TcName { get; set; }
public int TcNumber { get; set; }
public virtual ICollection<Student> Students { get; set; } //班级里面有很多学生
}
    /// <summary>
/// 学生表
/// </summary>
public class Student
{
public Student()
{
this.Courses=new Collection<Course>();
}
public Guid StudentId { get; set; }
public string SName { get; set; }
public int SAge { get; set; }
public Guid TheClassId { get; set; }
public virtual TheClass TheClass { get; set; } //学生属于一个班级
public virtual ICollection<Course> Courses { get; set; }
}

Fluent API(1:n)

    public class TheClassMap:EntityTypeConfiguration<TheClass>
{
public TheClassMap()
{
ToTable("TheClass");
HasKey(x => x.TheClassId);
Property(x => x.TcName).HasMaxLength(128);
Property(x => x.TcNumber);
HasMany(x => x.Students);
HasOptional(x => x.Monitor);
}
}
public class StudentMap : EntityTypeConfiguration<Student>
{
public StudentMap()
{
ToTable("Student");
HasKey(x => x.StudentId);
Property(x => x.SName).HasMaxLength(128);
HasRequired(x => x.TheClass).WithMany(x => x.Students).HasForeignKey(x => x.TheClassId);
HasMany(x => x.Courses).WithMany(x => x.Student);
}
}

数据库中的表

三:多对多联系(n:n)

A实体集中的每一个实体在实体集B中有n个实体与之对应(n>=0),反过来,B实体集中每一个实体在实体集A中有m个实体与之对应,这种关系就是多对多联系。

eg:学生和选修课。(一个学生可选修多门课程,一门课程可由多名学生选修),订单和产品(一个订单里面有多个产品,一个产品可在多个订单中)。

属性模型

    /// <summary>
/// 课程类
/// </summary>
public class Course
{
public Course()
{
this.Student=new Collection<Student>();
}
public Guid CourseId { get; set; }
public string CName { get; set; }
public virtual ICollection<Student> Student { get; set; }
}
    /// <summary>
/// 学生表
/// </summary>
public class Student
{
public Student()
{
this.Courses=new Collection<Course>();
}
public Guid StudentId { get; set; }
public string SName { get; set; }
public int SAge { get; set; }
public Guid TheClassId { get; set; }
public virtual TheClass TheClass { get; set; } //学生属于一个班级
public virtual ICollection<Course> Courses { get; set; }
}

Fluent API

    public class CourseMap:EntityTypeConfiguration<Course>
{
public CourseMap()
{
ToTable("Course");
HasKey(x => x.CId);
Property(x => x.CName).HasMaxLength(128);
HasMany(x => x.Students).WithMany(x=>x.Courses).Map(m =>
{
m.ToTable("CourseStudentId");
m.MapLeftKey("CourseId");
m.MapRightKey("StudentId");
}); //这里设置多对多关系中间表的表名和列名
}
}
    public class StudentMap:EntityTypeConfiguration<Student>
{
public StudentMap()
{
ToTable("Student");
HasKey(x => x.SId);
Property(x => x.SName).HasMaxLength(128);
HasRequired(x => x.TheClass).WithMany(x=>x.Students).HasForeignKey(x=>x.TheClassId);
HasMany(x => x.Courses).WithMany(x=>x.Students);
}
}

数据库中的表

在多对多关系中,我们可以设置中间表的表名和列名,就如同上面那样子。

上面就是数据库表的三中关系,其中里面的一对一是一对多的特例,一对多是多对多的特例,我们只有掌握好这些最基本的知识,才能理解更加复杂表中的关系。

出现错误

*:这是我在使用数据迁移的时候出现的错误。原因是我前面的主键设置的是(SId,CId,TCId,MId).EF在生成数据库的时候报错,我们应该把他们设置为(StudentId,CourseId,TheClassId,MonitorId),这样子就会出现我们的表。

* 上面的错误很大一定程度是因为我没有重写EF里面的OnModelCreating类导致的,我前面多对多表创建时出现的问题也是由于这个原因导致的。

下面是数据上下文中的代码

    public class BlogDbContent : DbContext, IDisposable
{
public BlogDbContent()
: base("name=BlogsEntities")
{
} public DbSet<BlogUser> BlogUser { get; set; }
public DbSet<Post> Post { get; set; }
public DbSet<Course> Course { get; set; }
public DbSet<Student> Students { get; set;}
public DbSet<TheClass> TheClass { get; set; }
public DbSet<Monitor> Monitor { get; set; } /// <summary>
/// 重写配置类
/// </summary>
/// <param name="modelBuilder"></param>
protected override void OnModelCreating(DbModelBuilder modelBuilder)
{
var typesToRegister = Assembly.GetExecutingAssembly().GetTypes()
.Where(type => !String.IsNullOrEmpty(type.Namespace))
.Where(type => type.BaseType != null && type.BaseType.IsGenericType && type.BaseType.GetGenericTypeDefinition() == typeof(EntityTypeConfiguration<>));
foreach (var type in typesToRegister)
{
dynamic configurationInstance = Activator.CreateInstance(type);
modelBuilder.Configurations.Add(configurationInstance);
}
base.OnModelCreating(modelBuilder);
}
}

数据库的连接字符串<App.Config>

  <connectionStrings>
<add name="BlogsEntities" connectionString="Data Source=localhost;Initial Catalog=Ahui_Blog;Integrated Security=False;
Persist Security Info=False;User ID=sa;Password=******" providerName="System.Data.SqlClient" />
</connectionStrings>

附件:

Fluent API参考

http://www.cnblogs.com/oppoic/p/ef_one-to-one_one-to-many_many-to-many_cascadedelete.html

http://www.cnblogs.com/panchunting/p/entity-framework-code-first-fluent-api-configuring-relationships.html

理解一对一关系

http://www.cnblogs.com/dudu/archive/2011/07/08/entity_framework_one_to_one_two_way.html

DB表的关系及EF中Fluent API的使用的更多相关文章

  1. EF使用Fluent API配置映射关系

    定义一个继承自EntityTypeConfiguration<>泛型类的类来定义domain中每个类的数据库配置,在这个自定义类的构造函数中使用我们上次提到的那些方法配置数据库的映射. 映 ...

  2. EF:Fluent API 把一对多映射为一对一

    假设有两张表:A表和B表.A表与B表在数据库中的关系是一对多,但我们需要在EF中映射为一对一. 首先在A实体类和B实体类中互相为对方增加一个实体类的属性: public A { public B B ...

  3. 第六节:框架搭建之EF的Fluent Api模式的使用流程

    一. 前言 沉寂了约一个月的时间,今天用一篇简单的文章重新回归博客,主要来探讨一下Fluent Api模式在实际项目中的使用流程. 1. Fluent API属于EF CodeFirst模式的一种,E ...

  4. EF Core Fluent API

    多对多配置 先安装 Install-Package MySql.Data.EntityFrameworkCore 创建3个表 创建类 public class Role { public long I ...

  5. 第十八篇 .NET高级技术之Linq与EF Code-First Fluent API基础讲解

    1.FluentApi简介 在这里提供了一个fluentapi基础的DEMO然后咱们在进一步的学习,直接上干货. 第一步在数据库创建一个表:person 第二步:新建控制台程序FluentAPI 第三 ...

  6. EF Core中Fluent Api如何删除指定数据表中的行

    这两天一直在研究在code first下如何删除数据表中的指定行,于是开始搜狗,后来百度,压根就找不到资料,后来一想可能我的搜索关键字有问题,而且ef core命令与ef的命令差不多,于是从这两个方面 ...

  7. 扩展EF的Fluent API中的 OnModelCreating方法 实现全局数据过滤器

    1.生成过滤的表达式目录树 protected virtual Expression<Func<TEntity, bool>> CreateFilterExpression&l ...

  8. 1.【使用EF Code-First方式和Fluent API来探讨EF中的关系】

    原文链接:http://www.c-sharpcorner.com/UploadFile/3d39b4/relationship-in-entity-framework-using-code-firs ...

  9. Entity Framework 实体框架的形成之旅--Code First模式中使用 Fluent API 配置(6)

    在前面的随笔<Entity Framework 实体框架的形成之旅--Code First的框架设计(5)>里介绍了基于Code First模式的实体框架的经验,这种方式自动处理出来的模式 ...

随机推荐

  1. 写了一个简单的NodeJS实现的进程间通信的例子

    1. cluster介绍 大家都知道nodejs是一个单进程单线程的服务器引擎,不管有多么的强大硬件,只能利用到单个CPU进行计算.所以,有人开发了第三方的cluster,让node可以利用多核CPU ...

  2. 人人都是 DBA(XIV)存储过程信息收集脚本汇编

    什么?有个 SQL 执行了 8 秒! 哪里出了问题?臣妾不知道啊,得找 DBA 啊. DBA 人呢?离职了!!擦!!! 程序员在无处寻求帮助时,就得想办法自救,努力让自己变成 "伪 DBA& ...

  3. Aoite 系列(01) - 比 Dapper 更好用的 ORM

    Aoite 是一个适于任何 .Net Framework 4.0+ 项目的快速开发整体解决方案.Aoite.Data 适用于市面上大多数的数据库提供程序,通过统一封装,可以在日常开发中简单便捷的操作数 ...

  4. UWP 统一平台开发介绍

    什么是UWP? 很多程序员都有一个梦想:希望自己开发的软件能够轻而易举的在所有平台上运行,而不是把同样的需求,用不同的技术.工具重新开发才能够运行在所有平台上.这就是跨平台,很多软件从业者都在为这个梦 ...

  5. Spring Trasnaction管理(1)- 线程间事务隔离

    问题导读 Spring中事务是如何实现的 Spring中各个线程间是如何进行连接.事务隔离的 Spring事务配置 Spring的事务管理应该是日常开发中总会碰到的,但是Spring具体是怎么实现线程 ...

  6. CocoaPods 深入使用

    在 CocoaPods 使用中介绍了基本的使用 写项目的时候想用到 SQLite.swift第三方库,但是问题来了 pod search SQLite.swift  //执行这条语句,搜索不到结果 但 ...

  7. css 常见时间轴的做法(————————————————时间轴——————————————————)

    <!DOCTYPE html> <html> <head> <meta charset="UTF-8"> <title> ...

  8. 60,000毫秒内对Linux的性能诊断效的方法

    转载于:http://www.itxuexiwang.com/a/liunxjishu/2016/0225/168.html?1456484140 60,000 毫秒内对 Linux 的性能诊断 当你 ...

  9. Js~(function(){})匿名自执行方法的作用

    匿名自执行方法体(function(){})经常用在设计JS插件上面,它定义相关组件的行为,自动初始化相关属性,而且在页面中可以直接执行,你不需要手动执行它,它被自动被执行! 在设计你的匿名自执行方法 ...

  10. Markdown入门基础

    // Markdown入门基础 最近准备开始强迫自己写博文,以治疗严重的拖延症,再不治疗就“病入骨髓,司命之所属,无奈何”了啊.正所谓“工欲善其事,必先利其器”,于是乎在写博文前,博主特地研究了下博文 ...