Entity Framework——建模建库
1数据库初始化策略选择
三种初始化策略:
1)CreateDatabaseIfNotExists:默认的策略。如果数据库不存在,那么就创建数据库。但是如果数据库已存在,而且实体发生了变化,就会出现异常。
2)DropCreateDatabaseIfModelChanges:模型改变时,原来的数据库会被删除,自动重新创建一个新的数据库。
3)DropCreateDatabaseAlways:每次运行都会删除原来的数据库,然后重新生成数据库。
4)Null:在Codefirst模式下,当实体结构改变时,运行程序不会自动生成表,改变实体结构与改变表结构互不影响,
前三种策略无法应对的问题是:分别改变实体模型和数据库表结构。即,当使用CreateDatabaseIfNotExists策略时,修改实体模型会抛异常(The model backing the <Database> context has changed since the database was created
);若使用DropCreateDatabaseIfModelChanges模式,那么每次运行都会重新生成数据库,这导致历史数据丢失,然而提前备份数据这种策略比较麻烦,尤其是系统上线以后。
解决办法:
采用第四种初始化策略,初次使用codefirst方式创建好数据库以后,不使用任何数据库初始化策略,即给Database.SetInitializer传null。
- [DbConfigurationType(typeof(MySqlEFConfiguration))]
- public class HY_WebApiContext : DbContext
- {
- public HY_WebApiContext(): base("name=HY_WebApiContext")
- {
- Database.SetInitializer<HY_WebApiContext>(null);
- }
- ......
- }
2实体关系与依赖默认规则创建的表关系
Codefirst模式下,实体与表之间的映射,随实体关系的不同而不同。
1)实体之间为一对一关系
实体
- public class EntityOne
- {
- public int Id { get; set; }
- public string FieldEntityOne { get; set; }
- public EntityTwo EntityTwo { get; set; }
- }
- public class EntityTwo
- {
- public int Id { get; set; }
- public string FieldEntityTwo { get; set; }
- }
表结构
entityones,其中EntityTwo_Id为外键。
EntityTwo
2)实体间的一对多关系
实体
- public class EntityOne
- {
- public int Id { get; set; }
- public string FieldEntityOne { get; set; }
- public List<EntityTwo> EntityTwos { get; set; }
- }
- public class EntityTwo
- {
- public int Id { get; set; }
- public string FieldEntityTwo { get; set; }
- }
表结构
entityones
entitytwoes,其中EntityOne_Id是外键
3)实体间的多对多关系
实体
- public class EntityOne
- {
- public int Id { get; set; }
- public string FieldEntityOne { get; set; }
- public List<EntityTwo> EntityTwos { get; set; }
- }
- public class EntityTwo
- {
- public int Id { get; set; }
- public string FieldEntityTwo { get; set; }
- public List<EntityOne> EntityOnes { get; set; }
- }
表结构
Codefirst模式下,两个实体间的多对多关系,映射为三张表,其中一张表表示实体间的联系。
entityones
entitytwoes
entitytwoentityones,其中EntityTwo_Id和EntityOne_Id是外键,这两个外键构成了改表的复合主键。
4)实体包含类型相同的两个或多个名称不同的导航属性
实体
- public class EntityOne
- {
- public int Id { get; set; }
- public string FieldEntityOne { get; set; }
- public EntityTwo EntityTwos { get; set; }
- public EntityTwo EntityTwosOther { get; set; }
- }
- public class EntityTwo
- {
- public int Id { get; set; }
- public string FieldEntityTwo { get; set; }
- }
表
Entityones,其中EntityTwos_Id、EntityTwosOther_Id是外键。
entitytwoes
3为关系创建实体:
一个用户可以订阅多种出版物,一种出版物可被多个用户订阅,实体建模如下:
- public class Publication
- {
- public int Id { get; set; }
- public virtual ICollection<User> Users { get; set; }
- ......
- }
- public class User
- {
- public int Id { get; set; }
- public virtual ICollection<Publication> Publications { get; set; }
- ......
- }
EF框架对上述多对多关系的默认处理方式为生成三张表:publications,user,publicationusers
其中publicationusers为关系表,表的属性只包括两个表的主键。
问题1:publications表的数据会大量重复:假设用户A订阅了电子学报,publications表里会有一条关于电子学报的记录,当用户B也订阅电子学报的时候,又会将这条数据插入publications表,如此等等。
解决方案:
每次向publications表插入记录时,先在表中查找待插入的刊物是否存在,如果存在就不插入,只更新publicationusers表。
问题2:虽然使用上面的方法可以解决这个问题,但用户何时订阅了一种刊物,这类信息没有被记录下来。
解决方案:
添加一个实体,表达publications,user这两个实体之间的关系,实体如下:
- public class Publication
- {
- public int Id { get; set; }
- ......
- }
- public class User
- {
- public int Id { get; set; }
- ......
- }
- public class PublicationUser
- {
- public int Id { get; set; }
- /// <summary>
- /// 出版物
- /// </summary>
- public Publication Publication { get; set; }
- /// <summary>
- /// 所属用户
- /// </summary>
- public User User { get; set; }
- /// <summary>
- /// 记录插入时间
- /// </summary>
- public DateTime InsertTime { get; set; }
- /// <summary>
- /// 记录修改时间
- /// </summary>
- public DateTime UpdateTime { get; set; }
- }
经上述处理后,EF仍然生成三张表publications,user,publicationusers,与之前不同的是publicationusers表中多了Id ,InsertTime ,UpdateTime 这三个字段,同时去掉了publications,user其中的导航属性,这样的设计满足需求,且无冗余。那么经过这样的修改后,每一个PublicationUser实例对应了表中的一条记录。
Entity Framework——建模建库的更多相关文章
- 查询大数据表的效率对比:Linq to SQL、Entity Framework、企业库存储过程、ADO.Net
最近因为要开发大数据量网站,特作比较. Linq to SQL 查询 记录数:399997Linq to SQL 查询 Milliseconds:1910视图查询 记录数:399997视图查询 Mil ...
- Entity Framework的扩展库
https://github.com/jcachat/EntityFramework.DynamicFilters Provides global & scoped filters for E ...
- Programming Entity Framework CodeFirst--数据库约定和配置
这一章主要主要讲的是我们的模型如何映射到数据库,而不影响模型,以及不同的映射场景. 一.表名和列名 1.指定表名 [Table("PersonPhotos")] public cl ...
- ADO.NET-EF:ADO.NET Entity Framework 百科
ylbtech-ADO.NET-EF:ADO.NET Entity Framework 百科 ADO.NET Entity Framework 是微软以 ADO.NET 为基础所发展出来的对象关系对应 ...
- 8天掌握EF的Code First开发之Entity Framework介绍
返回<8天掌握EF的Code First开发>总目录 本篇目录 Entity Framework概要 什么是ORM Entity Framework简史 Entity Framework具 ...
- 转载 8天掌握EF的Code First开发之Entity Framework介绍
转载原地址: http://www.cnblogs.com/farb/p/IntroductionToEF.html Entity Framework概要 Entity Framework是微软的O ...
- Entity Framework开发介绍
一.Entity Framework概要 Entity Framework是微软的Object Relational Mapper(对象关系映射),也就是我们平常说的ORM,它可以让应用程序开发者将关 ...
- 《Entity Framework 6 Recipes》中文翻译系列 (37) ------ 第六章 继承与建模高级应用之独立关联与外键关联
翻译的初衷以及为什么选择<Entity Framework 6 Recipes>来学习,请看本系列开篇 6-13 在基类中应用条件 问题 你想从一个已存在的模型中的实体派生一个新的实体, ...
- Entity Framework扩展库
这个Entity Framework扩展完全支持EF 5.0/6.0,项目地址 https://github.com/loresoft/EntityFramework.Extended,这个库支持批量 ...
随机推荐
- Java基础总结--方法(函数)
---函数的作用---实现特定功能的代码--是一种代码重用的方式---函数的格式---访问修饰符 返回值类型 函数名(参数列表){语句:} 参数列表包含参数的类型和参数名(参数列表要注意顺序)---关 ...
- python中openpyxl的用法【安装,以及一些基本的操作】
概述 Openpyxl是python中简单易用的操作excel电子表格的一个模块.接下来呢,跟博主一起学习一下吧 ----_<_>_---- 首先先清楚一些excel的基本概念: 在op ...
- 谈一谈原生JS中的【面向对象思想】
[重点提前说:面向对象的思想很重要!] 最近开始接触学习后台的PHP语言,在接触到PHP中的面向对象相关思想之后,突然想到之前曾接触的JS中的面向对象思想,无奈记性太差, ...
- jquery的2.0.3版本源码系列(7):3043行-3183行,deferred延迟对象,对异步的统一管理
目录 part1 deferred延迟对象 part2 when辅助方法 网盘源代码 链接: https://pan.baidu.com/s/1skAj8Jj 密码: izta part1 defe ...
- java基础解析系列(十)---ArrayList和LinkedList源码及使用分析
java基础解析系列(十)---ArrayList和LinkedList源码及使用分析 目录 java基础解析系列(一)---String.StringBuffer.StringBuilder jav ...
- 使用JS实现图片轮播滚动跑马灯效果
我的第一篇文章.哈哈.有点小鸡冻. 之前在百度搜索"图片轮播"."图片滚动",结果都是那种可以左右切换的.也是我们最常见的那种.可能是搜索 关键字的问题吧. ...
- hbase-1.2.5完全分布式部署
详细配置参考http://abloz.com/hbase/book.html#distributed ,2.2.2.2小节 1.修改hbase-site.xml文件.添加如下配置 cluster为ha ...
- ASP.NET没有魔法——ASP.NET Identity的加密与解密
前面文章介绍了如何使用Identity在ASP.NET MVC中实现用户的注册.登录以及身份验证.这些功能都是与用户信息安全相关的功能,数据安全的重要性永远放在第一位.那么对于注册和登录功能来说要把密 ...
- java语言将任意一个十进制数数字转换为二进制形式,并输出转换后的结果
package com.llh.demo; import java.util.Scanner; /** * * @author llh * */ public class Test { /* * 将任 ...
- Github和Github for windows的使用简介
很多程序员都把自己开发的代码放到Github上,方便自己管理也有利于别人查阅.所以这两天我也捣鼓了一下这个东西,现在把怎么使用Github和Github for windows简单的总结一下. 1.现 ...