为什么要改用code first

用过code first的基本上都不会再想用回model first或是db first(谁用谁知道)。不要问我为什么不一开始就直接使用code first,因为那个时候我还不会(甚至还把model first当成了code first)。

因为工作中使用的就是code first,且越用越习惯,越用越喜欢。

原因如果:

  • 再也用为每次生成那个笨重的edmx文件性急了
  • 再也不用当心保存tt文件而丢失特性、注销、扩展方法了
  • 再也不用为了使用微软的验证插件非得写Metadata文件了
  • 再也不用为了扩展tt文件生成的实体类去写(partial)部分类了。
  • 再也不用为了生成满足自己需要的实体而去修改那些坑爹的tt文件里面的语法代码了(如:默认每个实体继承一个父类)
  • 再也不用为了查找edmx文件打不开,去编辑庞大的edmx文件中找那些坑爹的错误了。
  • 等等还有些暂时没想到的....

说改就改

修改前实体:db first(由tt文件生成)

修改后实体:code first(完全手写)

然后把实体更新到数据库对应的表结构。执行命令Enable-Migrations

遇到问题:

The EntityFramework package is not installed on project ''.(原因:因为没有选择“默认项目”)

继续问题:

The project 'Blogs.Model' failed to build.(原因:没有建一个继承于DbContext的类)

ok,提示已经启用迁移。

然后我们执行命令:Add-Migration blogs

异常: 从数据库中获取提供程序信息时出错。这可能是 Entity Framework 使用的连接字符串不正确导致的。有关详细信息,请查看内部异常并确保连接字符串正确。

我的乖乖,我非常确定我们字符串链接是正确的啊。

最后确定忘记给数据连接上下文在构造函数中传入配置文件的数据库链接名。

  public BlogDbContext()
: base("HiBlogsTest")
{
}

再执行(Add-Migration blogs),再出错:

异常:无法加载指定的元数据资源。(百度之,原来是链接字符串有问题。http://www.cnblogs.com/chengxiaohui/articles/2106765.html

 <add name="HiBlogsTest" connectionString="metadata=res://*/Model1.csdl|res://*/Model1.ssdl|res://*/Model1.msl;provider=System.Data.SqlClient;
provider connection string=&quot;
data source=.;
initial catalog=HiBlogsTest;
user id=sa;
password=123qwe;
MultipleActiveResultSets=True;
App=EntityFramework&quot;" providerName="System.Data.EntityClient" />

改成:(那一堆csdl、ssdl、msl什么都不要了,就留个简单的链接。干净)

<add name="HiBlogsTest" connectionString="Data Source=.;Initial Catalog=HiBlogsTest;User ID=sa;Password=123qwe;" providerName="System.Data.SqlClient" />

ok,终于没有看见红色的字了。

且看到了一个自动生成的blogs文件。且不管,看看数据库是否有表结构。

空空如也。(屁都没看到一个)(原因:BlogDbContext上下文中没有添加实体,没有告诉程序要生成哪些实体到数据库)

给BlogDbContext类添加数据代码:

  public class BlogDbContext : DbContext
{
public BlogDbContext()
: base("HiBlogsTest")
{
}
public DbSet<BlogInfo> BlogInfos { get; set; }
public DbSet<BlogComment> BlogComments { get; set; }
public DbSet<BlogReadInfo> BlogReadInfos { get; set; }
public DbSet<BlogTag> BlogTags { get; set; }
public DbSet<BlogType> BlogTypes { get; set; }
public DbSet<BlogUser> BlogUsers { get; set; }
public DbSet<BlogUserInfo> BlogUserInfos { get; set; }
}

然后执行 :Add-Migration blogs 再执行 update-database

终于看到表数据了。

有了表还不行,我们还没有主外键。

修改BlogDbContext如下:

public class BlogDbContext : DbContext
{
public BlogDbContext()
: base("HiBlogsTest")
{
}
protected override void OnModelCreating(DbModelBuilder modelBuilder)
{
base.OnModelCreating(modelBuilder); var entityBlogUser = modelBuilder.Entity<BlogUser>(); entityBlogUser.HasMany(p => p.BlogInfos).WithRequired(t => t.BlogUser)
.Map(m => m.MapKey("BlogUserId")); entityBlogUser.HasRequired(p => p.BlogUserInfo).WithRequiredPrincipal(t => t.BlogUser)
.Map(m => m.MapKey("BlogUserId")); entityBlogUser.HasMany(p => p.BlogTags).WithRequired(t => t.BlogUser)
.Map(m => m.MapKey("BlogUserId")); entityBlogUser.HasMany(p => p.BlogTypes).WithRequired(t => t.BlogUser)
.Map(m => m.MapKey("BlogUserId")); entityBlogUser.HasMany(p => p.BlogComments).WithRequired(t => t.BlogUser)
.Map(m => m.MapKey("BlogUserId")); var entityBlogInfo = modelBuilder.Entity<BlogInfo>(); entityBlogInfo.HasMany(p => p.BlogTags).WithMany(t => t.BlogInfos)
.Map(m => m.ToTable("BlogInfo_BlogTag")); entityBlogInfo.HasMany(p => p.BlogTypes).WithMany(t => t.BlogInfos)
.Map(m => m.ToTable("BlogInfo_BlogType")); entityBlogInfo.HasMany(p => p.BlogComments).WithRequired(t => t.BlogInfo)
.Map(m => m.MapKey("BlogInfoId")); entityBlogInfo.HasMany(p => p.BlogReadInfos).WithRequired(t => t.BlogInfo)
.Map(m => m.MapKey("BlogInfoId"));
} public DbSet<BlogInfo> BlogInfos { get; set; }
public DbSet<BlogComment> BlogComments { get; set; }
public DbSet<BlogReadInfo> BlogReadInfos { get; set; }
public DbSet<BlogTag> BlogTags { get; set; }
public DbSet<BlogType> BlogTypes { get; set; }
public DbSet<BlogUser> BlogUsers { get; set; }
public DbSet<BlogUserInfo> BlogUserInfos { get; set; }
}

然后重新命令:Add-Migration blogs 再执行 update-database

又见错误:

将 FOREIGN KEY 约束 'FK_dbo.BlogInfo_dbo.BlogUser_BlogUserId' 引入表 'BlogInfo' 可能会导致循环或多重级联路径。请指定 ON DELETE NO ACTION 或 ON UPDATE NO ACTION,或修改其他 FOREIGN KEY 约束。
无法创建约束。请参阅前面的错误消息。

于是,一个一个的外键删掉,又一个个的来建。终于发现:(下图是数据库关系图,mssql生成的)

百度之:(原来是为了约束联级删除数据做的约束。实在话,还没玩过联级删除了,说明这个需求应该不是很常用。找个方法禁用可否?)

直接加一个.WillCascadeOnDelete(false)就可以了。(http://www.cnblogs.com/chear/archive/2012/11/09/2762145.html

public class BlogDbContext : DbContext
{
public BlogDbContext()
: base("HiBlogsTest")
{
}
protected override void OnModelCreating(DbModelBuilder modelBuilder)
{
base.OnModelCreating(modelBuilder); var entityBlogUser = modelBuilder.Entity<BlogUser>(); entityBlogUser.HasMany(p => p.BlogInfos).WithRequired(t => t.BlogUser)
.Map(m => m.MapKey("BlogUserId")).WillCascadeOnDelete(false);
//与上面等效
//modelBuilder.Entity<BlogInfo>().HasRequired(p => p.BlogUser).WithMany(t => t.BlogInfos) //以BlogUser为主表(BlogUserInfo为从表,建立外键)
entityBlogUser.HasRequired(p => p.BlogUserInfo).WithRequiredPrincipal(t => t.BlogUser)
.Map(m => m.MapKey("BlogUserId")).WillCascadeOnDelete(false);
//等效于HasRequired(p => ).WithOptional(i => ); ////以BlogUserInfo为主表(BlogUser为从表,建立外键)
//modelBuilder.Entity<BlogUser>().HasRequired(p => p.BlogUserInfo).WithRequiredDependent(t => t.BlogUser)
//.Map(m => m.MapKey("BlogUserId")).WillCascadeOnDelete(false);
//等效于 HasOptional(p => ).WithRequired(i => ); entityBlogUser.HasMany(p => p.BlogTags).WithRequired(t => t.BlogUser)
.Map(m => m.MapKey("BlogUserId")).WillCascadeOnDelete(false); entityBlogUser.HasMany(p => p.BlogTypes).WithRequired(t => t.BlogUser)
.Map(m => m.MapKey("BlogUserId")).WillCascadeOnDelete(false); entityBlogUser.HasMany(p => p.BlogComments).WithRequired(t => t.BlogUser)
.Map(m => m.MapKey("BlogUserId")).WillCascadeOnDelete(false); var entityBlogInfo = modelBuilder.Entity<BlogInfo>(); entityBlogInfo.HasMany(p => p.BlogTags).WithMany(t => t.BlogInfos)
.Map(m => m.ToTable("BlogInfo_BlogTag")); entityBlogInfo.HasMany(p => p.BlogTypes).WithMany(t => t.BlogInfos)
.Map(m => m.ToTable("BlogInfo_BlogType")); entityBlogInfo.HasMany(p => p.BlogComments).WithRequired(t => t.BlogInfo)
.Map(m => m.MapKey("BlogInfoId")).WillCascadeOnDelete(false); entityBlogInfo.HasMany(p => p.BlogReadInfos).WithRequired(t => t.BlogInfo)
.Map(m => m.MapKey("BlogInfoId")).WillCascadeOnDelete(false);
} public DbSet<BlogInfo> BlogInfos { get; set; }
public DbSet<BlogComment> BlogComments { get; set; }
public DbSet<BlogReadInfo> BlogReadInfos { get; set; }
public DbSet<BlogTag> BlogTags { get; set; }
public DbSet<BlogType> BlogTypes { get; set; }
public DbSet<BlogUser> BlogUsers { get; set; }
public DbSet<BlogUserInfo> BlogUserInfos { get; set; }
}

然后重新命令:Add-Migration blogs 再执行 update-database

完美,表结构过来了。表关系过来了。(接下来就是该代码了,因为表名做了小的改动,字段也做了少许调整所以改的东西还真不少。整整改了一天时间。)

现在回过头来想想,之前是先model first之后小许改动就用的db first。以前怎么没有遇到过(将 FOREIGN KEY 约束 'FK_dbo.BlogInfo_dbo.BlogUser_BlogUserId' 引入表 'BlogInfo' 可能会导致循环或多重级联路径。请指定 ON DELETE NO ACTION 或 ON UPDATE NO ACTION,或修改其他 FOREIGN KEY 约束。
无法创建约束。请参阅前面的错误消息。)这个错误。好奇心驱使,觉得看看以前的代码的edmx是怎么管理这种关系的。

很惊奇的发现,完全没有问题。于是,不死心看看数据库里面是不是有什么蹊跷。

搜噶,原来如此。通过model first生成的主外键关系默认就没有设计联级删除,而code first默认设置就是联级删除。

以上内容,都是我胡说八道。谢谢您的阅读,希望对您有那么一点点作用。

Hi-Blogs源码地址:http://git.oschina.net/zhaopeiym/Hi-Blogs

最近因为工作实在太慢,开源博客长久没有更新。今天突然来回翻了好几遍,发现半年前的自己写的代码是如此的不堪入目。

今天仅仅只是把db first改成了code first,发霉的代码我还得找个时间好好重构重构。

首发地址:http://www.cnblogs.com/zhaopei/p/5540532.html

一步步开发自己的博客 .NET版(9、从model first替换成code first 问题记录)的更多相关文章

  1. 一步步开发自己的博客 .NET版(1、基本显示)

    前言 我们每个猿都有一个搭建自己独立博客的梦,我也不例外.以前想 现在想 以后也想.之所以一直迟迟没有着手,是因为难以跨出第一步.每次心里想着,等我以后技术好了再说,然后就没有然后了.以前用过word ...

  2. 一步步开发自己的博客 .NET版(3、注册登录功能)

    前言 这次开发的博客主要功能或特点:    第一:可以兼容各终端,特别是手机端.    第二:到时会用到大量html5,炫啊.    第三:导入博客园的精华文章,并做分类.(不要封我)    第四:做 ...

  3. 一步步开发自己的博客 .NET版(4、文章发布功能)百度编辑器

    前言 这次开发的博客主要功能或特点: 第一:可以兼容各终端,特别是手机端. 第二:到时会用到大量html5,炫啊. 第三:导入博客园的精华文章,并做分类.(不要封我) 第四:做个插件,任何网站上的技术 ...

  4. 一步步开发自己的博客 .NET版(5、Lucenne.Net 和 必应站内搜索)

    前言 这次开发的博客主要功能或特点:    第一:可以兼容各终端,特别是手机端.    第二:到时会用到大量html5,炫啊.    第三:导入博客园的精华文章,并做分类.(不要封我)    第四:做 ...

  5. 一步步开发自己的博客 .NET版 剧终篇(6、响应式布局 和 自定义样式)

    前言 这次开发的博客主要功能或特点:    第一:可以兼容各终端,特别是手机端.    第二:到时会用到大量html5,炫啊.    第三:导入博客园的精华文章,并做分类.(不要封我)    第四:做 ...

  6. 一步步开发自己的博客 .NET版(11、Web.config文件的读取和修改)

    Web.config的读取 对于Web.config的读取大家都很属性了.平时我们用得比较多的就是appSettings节点下配置.如: 我们对应的代码是: = ConfigurationManage ...

  7. 一步步开发自己的博客 .NET版(10、前端对话框和消息框的实现)

    关于前端对话框.消息框的优秀插件多不胜数.造轮子是为了更好的使用轮子,并不是说自己造的轮子肯定好.所以,这个博客系统基本上都是自己实现的,包括日志记录.响应式布局.评论功能等等一些本可以使用插件的.好 ...

  8. ASP.NET 一步步开发自己的博客 .NET版(11、Web.config文件的读取和修改)

    原文:http://www.cnblogs.com/zhaopei/p/5677053.html

  9. 一步步搭建自己的博客 .NET版(2、评论功能)

    前言 这次开发的博客主要功能或特点:    第一:可以兼容各终端,特别是手机端.    第二:到时会用到大量html5,炫啊.    第三:导入博客园的精华文章,并做分类.(不要封我)    第四:做 ...

随机推荐

  1. ExtJS 4.2 介绍

    本篇介绍ExtJS相关知识,是以ExtJS4.2.1版本为基础进行说明,包括:ExtJS的特点.MVC模式.4.2.1GPL版本资源的下载和说明以及4种主题的演示. 目录 1. 介绍 1.1 说明 1 ...

  2. 关于 CSS 反射倒影的研究思考

    原文地址:https://css-tricks.com/state-css-reflections 译者:nzbin 友情提示:由于演示 demo 的兼容性,推荐火狐浏览.该文章篇幅较长,内容庞杂,有 ...

  3. [C#] C# 知识回顾 - 学会处理异常

    学会处理异常 你可以使用 try 块来对你觉得可能会出现异常的代码进行分区. 其中,与之关联的 catch 块可用于处理任何异常情况. 一个包含代码的 finally 块,无论 try 块中是否在运行 ...

  4. 在jekyll模板博客中添加网易云模块

    最近使用GitHub Pages + Jekyll 搭建了个人博客,作为一名重度音乐患者,博客里面可以不配图,但是不能不配音乐啊. 遂在博客里面引入了网易云模块,这里要感谢网易云的分享机制,对开发者非 ...

  5. 用WebRequest +HtmlAgilityPack 从外网抓取数据到本地

    相信大家对于WebRequest 并不陌生,我们在C#中发请求的方式,就是创建一个WebRequest .那么如果我们想发一个请求到外网,比如国内上不了的一些网站,那么该怎么做呢? 其实WebRequ ...

  6. Mybatis XML配置

    Mybatis常用带有禁用缓存的XML配置 <?xml version="1.0" encoding="UTF-8" ?> <!DOCTYPE ...

  7. 看图理解JWT如何用于单点登录

    单点登录是我比较喜欢的一个技术解决方案,一方面他能够提高产品使用的便利性,另一方面他分离了各个应用都需要的登录服务,对性能以及工作量都有好处.自从上次研究过JWT如何应用于会话管理,加之以前的项目中也 ...

  8. 信息安全-1:python之playfair密码算法详解[原创]

    转发注明出处: http://www.cnblogs.com/0zcl/p/6105825.html 一.基本概念 古典密码是基于字符替换的密码.加密技术有:Caesar(恺撒)密码.Vigenere ...

  9. Git(1)

    安装Git 完毕 (在开始菜单打开的话,打开的不是你想要的路径,切换路径很麻烦) 1.D盘新建 GitTest 文件夹 2.打开GitTest , 在空白的地方右键, 3.单击 Git Bash He ...

  10. docker4dotnet #3 在macOS上使用Visual Studio Code和Docker开发asp.net core和mysql应用

    .net猿遇到了小鲸鱼,觉得越来越兴奋.本来.net猿只是在透过家里那田子窗看外面的世界,但是看着海峡对岸的苹果园越来越茂盛,实在不想再去做一只宅猿了.于是,.net猿决定搭上小鲸鱼的渡轮到苹果园去看 ...