Entity Framework 6提供支持存储过程的新特性,本文具体演示Entity Framework 6 Code First的存储过程操作。

Code First的插入/修改/删除存储过程

  默认情况下下,Code First配置对全部实体的插入/修改/删除操作均直接针对表进行。从EF6开始可以配置对全部或部分实体来选择使用存储过程。

1. 基本实体映射

1.1 通过Fluent API,配置使用插入/修改/删除存储过程

modelBuilder
.Entity<Blog>()
.MapToStoredProcedures();

1.2 Code First在数据库中生成存储过程的约定

  ◊ 生成三个存储过程,名称分别为<type_name>_Insert, <type_name>_Update, <type_name>_Delete

  ◊ 参数名对应于属性名 (注意:如果在 property上使用 HasColumnName() 或者 Column attribute 来重命名,那么参数也将使用这个重命名过的名称 );

  ◊ The insert stored procedure 为每一个属性都有一个参数,除了那些标记为数据库产生的(identity or computed),返回结果为那些标记为数据库产生的属性列;

  ◊ The update stored procedure 为每一个属性都有一个参数,除了那些标记为数据库产生且模式为 computed 的。一些并发标记的需要一个代表原始值的参数。返回值为那些 computed property 的列;

  ◊ The delete stored procedure 参数为实体主键(或者组合主键),此外也需要为每一个独立关联的外键准备一个参数(指那些没有在实体上定义相应外键属性的关系),一些并发标记的需要一个代表原始值的参数。

  示例:

  实体文件blog.cs:

using System;
using System.Collections.Generic; namespace EF6.Models
{
public partial class Blog
{
public int BlogID { get; set; }
public string Name { get; set; }
public string Url { get; set; }
}
}

  实体映射文件BlogMap.cs:

using System.ComponentModel.DataAnnotations.Schema;
using System.Data.Entity.ModelConfiguration; namespace EF6.Models.Mapping
{
public class BlogMap : EntityTypeConfiguration<Blog>
{
public BlogMap()
{
// Primary Key
this.HasKey(t => t.BlogID); // Properties
this.Property(t => t.Name)
.HasMaxLength(); this.Property(t => t.Url)
.HasMaxLength(); // Table & Column Mappings
this.ToTable("Blog");
this.Property(t => t.BlogID).HasColumnName("BlogID");
this.Property(t => t.Name).HasColumnName("Name");
this.Property(t => t.Url).HasColumnName("Url"); // Procedures
this.MapToStoredProcedures();
}
}
}

  在程序包管理器控制台中依次执行:

PM> Enable-Migrations -EnableAutomaticMigrations
PM> Add-Migration InitialCreate
PM> Update-Database -Verbose

  执行完成之后生成数据库:

  监控生成数据的SQL语句:

CREATE TABLE [dbo].[Blog] (
[BlogID] [int] NOT NULL IDENTITY,
[Name] [nvarchar](50),
[Url] [nvarchar](100),
CONSTRAINT [PK_dbo.Blog] PRIMARY KEY ([BlogID])
CREATE PROCEDURE [dbo].[Blog_Insert]
@Name [nvarchar](50),
@Url [nvarchar](100)
AS
BEGIN
INSERT [dbo].[Blog]([Name], [Url])
VALUES (@Name, @Url) DECLARE @BlogID int
SELECT @BlogID = [BlogID]
FROM [dbo].[Blog]
WHERE @@ROWCOUNT > 0 AND [BlogID] = scope_identity() SELECT t0.[BlogID]
FROM [dbo].[Blog] AS t0
WHERE @@ROWCOUNT > 0 AND t0.[BlogID] = @BlogID
END
CREATE PROCEDURE [dbo].[Blog_Update]
@BlogID [int],
@Name [nvarchar](50),
@Url [nvarchar](100)
AS
BEGIN
UPDATE [dbo].[Blog]
SET [Name] = @Name, [Url] = @Url
WHERE ([BlogID] = @BlogID)
END
CREATE PROCEDURE [dbo].[Blog_Delete]
@BlogID [int]
AS
BEGIN
DELETE [dbo].[Blog]
WHERE ([BlogID] = @BlogID)
END

  2、重新默认约定

  2.1、重命名存储过程名称

modelBuilder
.Entity<Blog>()
.MapToStoredProcedures(s =>
s.Update(u => u.HasName("Modify_Blog")));
modelBuilder
.Entity<Blog>()
.MapToStoredProcedures(s =>
s.Update(u => u.HasName("Update_Blog"))
.Delete(d => d.HasName("Delete_Blog"))
.Insert(i => i.HasName("Insert_Bblog")));
modelBuilder
.Entity<Blog>()
.MapToStoredProcedures(s =>
{
s.Update(u => u.HasName("Update_Blog"));
s.Delete(d => d.HasName("Delete_Blog"));
s.Insert(i => i.HasName("Insert_Blog"));
});
using System.ComponentModel.DataAnnotations.Schema;
using System.Data.Entity.ModelConfiguration; namespace EF6.Models.Mapping
{
public class BlogMap : EntityTypeConfiguration<Blog>
{
public BlogMap()
{
// Primary Key
this.HasKey(t => t.BlogID); // Properties
this.Property(t => t.Name)
.HasMaxLength(); this.Property(t => t.Url)
.HasMaxLength(); // Table & Column Mappings
this.ToTable("Blog");
this.Property(t => t.BlogID).HasColumnName("BlogID");
this.Property(t => t.Name).HasColumnName("Name");
this.Property(t => t.Url).HasColumnName("Url"); // Procedures
this.MapToStoredProcedures(s =>{
s.Insert(u => u.HasName("Insert_Blog"));
s.Update(u => u.HasName("Update_Blog"));
s.Delete(u => u.HasName("Delete_Blog"));
});
}
}
}

  运行之后执行的SQL语句:

EXECUTE sp_rename @objname = N'dbo.Blog_Insert', @newname = N'Insert_Blog', @objtype = N'OBJECT'
EXECUTE sp_rename @objname = N'dbo.Blog_Update', @newname = N'Update_Blog', @objtype = N'OBJECT'
EXECUTE sp_rename @objname = N'dbo.Blog_Delete', @newname = N'Delete_Blog', @objtype = N'OBJECT'

  2.2、重命名存储过程参数名称

modelBuilder
.Entity<Blog>()
.MapToStoredProcedures(s =>
s.Update(u => u.Parameter(b => b.BlogID, "Blog_ID")));
using System.ComponentModel.DataAnnotations.Schema;
using System.Data.Entity.ModelConfiguration; namespace EF6.Models.Mapping
{
public class BlogMap : EntityTypeConfiguration<Blog>
{
public BlogMap()
{
// Primary Key
this.HasKey(t => t.BlogID); // Properties
this.Property(t => t.Name)
.HasMaxLength(); this.Property(t => t.Url)
.HasMaxLength(); // Table & Column Mappings
this.ToTable("Blog");
this.Property(t => t.BlogID).HasColumnName("BlogID");
this.Property(t => t.Name).HasColumnName("Name");
this.Property(t => t.Url).HasColumnName("Url"); // Procedures
this.MapToStoredProcedures(s =>
{
s.Update(u => u.Parameter(b => b.BlogID, "Blog_ID"));
});
}
}
}

  运行之后执行的SQL语句:

ALTER PROCEDURE [dbo].[Blog_Update]
@Blog_ID [int],
@Name [nvarchar](50),
@Url [nvarchar](100)
AS
BEGIN
UPDATE [dbo].[Blog]
SET [Name] = @Name, [Url] = @Url
WHERE ([BlogID] = @Blog_ID)
END
modelBuilder
.Entity<Blog>()
.MapToStoredProcedures(s =>
s.Update(u => u.HasName("Update_Blog")
.Parameter(b => b.BlogID, "Blog_ID")
.Parameter(b => b.Name, "Blog_Name")
.Parameter(b => b.Url, "Blog_Url"))
.Delete(d => d.HasName("Delete_Blog")
.Parameter(b => b.BlogID, "Blog_ID"))
.Insert(i => i.HasName("Insert_Blog")
.Parameter(b => b.Name, "Blog_Name")
.Parameter(b => b.Url, "Blog_Url")));

  2.3、重命名数据库自动生成列的返回值的列名

modelBuilder
.Entity<Blog>()
.MapToStoredProcedures(s =>
s.Insert(i => i.Result(b => b.BlogID, "generated_blog_identity")));
using System.ComponentModel.DataAnnotations.Schema;
using System.Data.Entity.ModelConfiguration; namespace EF6.Models.Mapping
{
public class BlogMap : EntityTypeConfiguration<Blog>
{
public BlogMap()
{
// Primary Key
this.HasKey(t => t.BlogID); // Properties
this.Property(t => t.Name)
.HasMaxLength(); this.Property(t => t.Url)
.HasMaxLength(); // Table & Column Mappings
this.ToTable("Blog");
this.Property(t => t.BlogID).HasColumnName("BlogID");
this.Property(t => t.Name).HasColumnName("Name");
this.Property(t => t.Url).HasColumnName("Url"); // Procedures
this.MapToStoredProcedures(s =>
{
s.Insert(i => i.Result(b => b.BlogID, "generated_blog_identity"));
});
}
}
}

  运行之后执行的SQL语句:

ALTER PROCEDURE [dbo].[Blog_Insert]
@Name [nvarchar](50),
@Url [nvarchar](100)
AS
BEGIN
INSERT [dbo].[Blog]([Name], [Url])
VALUES (@Name, @Url) DECLARE @BlogID int
SELECT @BlogID = [BlogID]
FROM [dbo].[Blog]
WHERE @@ROWCOUNT > 0 AND [BlogID] = scope_identity() SELECT t0.[BlogID] AS generated_blog_identity
FROM [dbo].[Blog] AS t0
WHERE @@ROWCOUNT > 0 AND t0.[BlogID] = @BlogID
END

  3、多对多关系

  两个多对多的实体类:

using System;
using System.Collections.Generic; namespace EF6.Models
{
public partial class Post
{
public int PostID { get; set; }
public string Title { get; set; }
public string Content { get; set; }
public virtual ICollection<Tag> Tags { get; set; }
}
}
using System;
using System.Collections.Generic; namespace EF6.Models
{
public partial class Tag
{
public int TagID { get; set; }
public string TagName { get; set; }
public virtual ICollection<Post> Posts { get; set; }
}
}

  映射存储过程:

protected override void OnModelCreating(DbModelBuilder modelBuilder)
{
modelBuilder.Entity<Post>()
.HasMany(p => p.Tags)
.WithMany(t => t.Posts)
.MapToStoredProcedures();
}

  默认生成的存储过程:

  ◊ 生成两个存储过程,命名为 <type_one><type_two>_Insert 和 <type_one><type_two>_Delete

  ◊ 参数为每一类型的主键(或组合主键),命名为 <type_name>_<property_name>

  代码运行后生成的存储过程:

CREATE PROCEDURE [dbo].[PostTag_Insert]
@Post_PostID [int],
@Tag_TagID [int]
AS
BEGIN
INSERT [dbo].[PostTags]([Post_PostID], [Tag_TagID])
VALUES (@Post_PostID, @Tag_TagID)
END
CREATE PROCEDURE [dbo].[PostTag_Delete]
@Post_PostID [int],
@Tag_TagID [int]
AS
BEGIN
DELETE [dbo].[PostTags]
WHERE (([Post_PostID] = @Post_PostID) AND ([Tag_TagID] = @Tag_TagID))
END

  参考资料:

  http://msdn.microsoft.com/en-us/data/dn468673

Entity Framework 6 Code First新特性:支持存储过程的更多相关文章

  1. Entity Framework Core 2.0 新特性

    本文翻译来自:https://docs.microsoft.com/en-us/ef/core/what-is-new/index 一.模型级查询过滤器(Model-level query filte ...

  2. 在Entity Framework 中用 Code First 创建新的数据库

    在Entity Framework 中用 Code First 创建新的数据库 (原文链接) 本文将逐步介绍怎样用Code First 创建新数据库,使用在代码中定义类和API中提供的特性(Attri ...

  3. AppBox升级进行时 - 拥抱Entity Framework的Code First开发模式

    AppBox 是基于 FineUI 的通用权限管理框架,包括用户管理.职称管理.部门管理.角色管理.角色权限管理等模块. 从Subsonic到Entity Framework Subsonic最早发布 ...

  4. Entity Framework 之 Code First

    使用NuGet助您玩转代码生成数据————Entity Framework 之 Code First [前言] 如果是Code First老鸟或者对Entity Framework不感兴趣,就不用浪费 ...

  5. 创建ASP.NET Core MVC应用程序(3)-基于Entity Framework Core(Code First)创建MySQL数据库表

    创建ASP.NET Core MVC应用程序(3)-基于Entity Framework Core(Code First)创建MySQL数据库表 创建数据模型类(POCO类) 在Models文件夹下添 ...

  6. Entity Framework Core Code First 项目实践

    Entity Framework Core Code First 实践 任何一种技术的出现都是为了解决一系列特定的问题,只有了解了技术所要解决的关键问题,才能理解它的真正用途,之后,才能在实践中用好它 ...

  7. 【极力分享】[C#/.NET]Entity Framework(EF) Code First 多对多关系的实体增,删,改,查操作全程详细示例【转载自https://segmentfault.com/a/1190000004152660】

      [C#/.NET]Entity Framework(EF) Code First 多对多关系的实体增,删,改,查操作全程详细示例 本文我们来学习一下在Entity Framework中使用Cont ...

  8. Entity Framework 第三篇 实体特性声明

    Entity Framework中对实体的特性声明有着严格的要求 1.实体必须要有主键特性,但是如果实体没有主键特性那怎么办? public int ExecuteSqlCommand(string ...

  9. MVC2、MVC3、MVC4、MVC5之间的区别 以及Entity Framework 6 Code First using MVC 5官方介绍教程

    现在MVC的技术日趋成熟,面对着不同版本的MVC大家不免有所迷惑 -- 它们之间有什么不同呢?下面我把我搜集的信息汇总一下,以便大家能更好的认识不同版本MVC的功能,也便于自己查阅. View Eng ...

随机推荐

  1. linpe包-让发送和接收数据分析更快和更容易

    1.简介 通常在R中从来进行分析和展现的数据都是以基本的格式保存的,如.csv或者.Rdata,然后使用.Rmd文件来进行分析的呈现.通过这个方式,分析师不仅可以呈现他们的统计分析的结果,还可以直接生 ...

  2. Python for Infomatics 第13章 网页服务三(译)

    注:文章原文为Dr. Charles Severance 的 <Python for Informatics>.文中代码用3.4版改写,并在本机测试通过. 13.6 应用程序接口API 现 ...

  3. android 从 phonegap 到 js webview 交互

    像生活类.办公协同类. 动态添加,下载等. 1.phonegap 我这里用了旧的版本,可能新版本变化大了. 创建asset资源文件夹,然后新建index.html copy 相应的js 文件进来. 创 ...

  4. 使用IHTMLDocument2解决弹出"为了让该网站给你提供个人化信息,是否允许在你计算机放置cookie?"

    mshtml可以说是一个不错的解析html利器,对于像我这样一直都是不用webbrowser,直接用socket或者WebRequest进行HTTP通讯 然后再用IHTMLDocument2.writ ...

  5. 弱省互测#1 t3

    题意 给出一棵n个点的树,求包含1号点的第k小的连通块权值和.(\(n<=10^5\)) 分析 k小一般考虑堆... 题解 堆中关键字为\(s(x)+min(a)\),其中\(s(x)\)表示\ ...

  6. 一个flex buider 3 在eclipse下不能编译的问题解决

    今天处理一个遗留的项目:项目使用了flex作为界面,装好flex Builder 3 并添加插件到eclipse,eclipse使用3.7版本. 导入项目,编译,发现编译时候出现 Errors run ...

  7. [转]android:动态创建多个按钮 及 批量设置监听

    之前投机取巧,先创建好多个按钮,再根据需要的数量进行部分隐藏,不过还是逃不过呀. 这样根本无法批量地 findId,批量地 设置监听. 所以今天还是认认真真地研究回“动态创建按钮”,终于,通过不断尝试 ...

  8. 为什么commonjs不适合于浏览器端

    有了服务器端模块以后,很自然地,大家就想要客户端模块.而且最好两者能够兼容,一个模块不用修改,在服务器和浏览器都可以运行. 但是,由于一个重大的局限,使得CommonJS规范不适用于浏览器环境.还是上 ...

  9. 部署Thomas Kyte 的 runstats 工具

    runstats是由Thomas Kyte开发的脚本,该脚本能对做同一件事的两个不同方法进行比较,得出孰优孰劣的结果. 1.授权 SQL> grant select on v_$statname ...

  10. SQL编程篇 (二) 定义与流程控制

    分类: sql编程:标准的sql 编程 * 纯sql 在标准的编程中又分为 sqlserver-->T-sql oracle-->pl-sql(扩展) 变量:在使用变量之前先定义 声明变量 ...