10-9. 在多对多关系中为插入和删除使用存储过程

问题

想要在一个无载荷的多对多关系中使用存储过程(存储过程只影响关系的连接表)

解决方案

假设有一个多对多关系的作者( Author)表和书籍( Book)表. 用连接表AuthorBook来做多对多关系,如 Figure 10-11.所示:

Figure 10-11. A payload-free, many-to-many relationship between an Author and a Book

当把表生成模型,那么模型就如Figure 10-12所示:

Figure 10-12. The model created by importing the tables in Figure 10-11

接下来用存储过程创建插入和删除操作:

1.在数据库中,创建如Listing 10-23所示的存储过程.

Listing 10-23. The stored Procedures for the Insert and Delete Actions

create procedure [chapter10].[InsertAuthorBook]

(@AuthorId int,@BookId int)

as

begin

insert into chapter10.AuthorBook(AuthorId,BookId) values (@AuthorId,@BookId)

end

go

create procedure [chapter10].[DeleteAuthorBook]

(@AuthorId int,@BookId int)

as

begin

delete chapter10.AuthorBook where AuthorId = @AuthorId and BookId = @BookId

end

2. 右击模型设计视图,选择“从数据库中更新模型”,选择 Listing 10-23所创建的存储过程,单击“完成”,这样就把存储过程添加到了模型里。

3. 目前版本的EF没有为一个关系插入和删除操作映射的设计视图,只能用手工去映射, 右击.edmx 文件,选择“打开方式”,选择“XML (文本)编辑器”. 在<AssociationSetMapping> 标签下插入 Listing 10-24所示的代码(译注:根据自己的例子命名修改代码里的)

Listing 10-24. Mapping the Stored Procedures to the Insert and Delete Actions for the Many-to-Many Association

<ModificationFunctionMapping>

<InsertFunction FunctionName="EFRecipesModel1009.Store.InsertAuthorBook">

<EndProperty Name="Author">

<ScalarProperty Name="AuthorId" ParameterName="AuthorId" />

</EndProperty>

<EndProperty Name="Book">

<ScalarProperty Name="BookId" ParameterName="BookId" />

</EndProperty>

</InsertFunction>

<DeleteFunction FunctionName="EFRecipesModel1009.Store.DeleteAuthorBook">

<EndProperty Name="Author">

<ScalarProperty Name="AuthorId" ParameterName="AuthorId" />

</EndProperty>

<EndProperty Name="Book">

<ScalarProperty Name="BookId" ParameterName="BookId" />

</EndProperty>

</DeleteFunction>

</ModificationFunctionMapping>

接下来Listing 10-25 代码演示了插入和删除操作.你可以用Sql Profiler来查看当

InsertAuthorBook 和DeleteAuthorBook 存储过程被EF在更新多对多关系时调用后生成的SQL语句

Listing 10-25. Inserting into the Model

class Program

{

static void Main(string[] args)

{

using (var context = new EFRecipesEntities1009())

{

context.Database.ExecuteSqlCommand("delete from chapter10.AuthorBook");

context.Database.ExecuteSqlCommand("delete from chapter10.book");

context.Database.ExecuteSqlCommand("delete from chapter10.Author");

var auth1 = new Author { Name = "Jane Austin" };

var book1 = new Book

{

Title = "Pride and Prejudice",

ISBN = "1848373104"

};

var book2 = new Book

{

Title = "Sense and Sensibility",

ISBN = "1440469563"

};

auth1.Books.Add(book1);

auth1.Books.Add(book2);

var auth2 = new Author { Name = "Audrey Niffenegger" };

var book3 = new Book

{

Title = "The Time Traveler's Wife",

ISBN = "015602943X"

};

auth2.Books.Add(book3);

context.Authors.Add(auth1);

context.Authors.Add(auth2);

context.SaveChanges();

context.Books.Remove(book1);

context.SaveChanges();

}

Console.WriteLine("\npress any key to exit...");

Console.ReadKey();

}

}

SQL Profiler里跟踪到的SQL语句如下(Listing 10-25)所示:

exec sp_executesql N'insert [Chapter10].[Author]([Name])values (@0)

select [AuthorId] from [Chapter10].[Author]

where @@ROWCOUNT > 0 and [AuthorId] = scope_identity()',N'@0 varchar(50)',@0='Jane Austin'

exec sp_executesql N'insert [Chapter10].[Author]([Name])values (@0)

select [AuthorId] from [Chapter10].[Author]

where @@ROWCOUNT > 0 and [AuthorId] = scope_identity()',N'@0 varchar(50)',

@0='Audrey Niffenegger'

exec sp_executesql N'insert [Chapter10].[Book]([Title], [ISBN])values (@0, @1)

select [BookId]    from [Chapter10].[Book]

where @@ROWCOUNT > 0 and [BookId] = scope_identity()',N'@0 varchar(50),

@1 varchar(50)',@0='Pride and Prejudice',@1='1848373104'

exec sp_executesql N'insert [Chapter10].[Book]([Title], [ISBN])values (@0, @1)

select [BookId] from [Chapter10].[Book]

where @@ROWCOUNT > 0 and [BookId] = scope_identity()',N'@0 varchar(50),

@1 varchar(50)',@0='Sense and Sensibility',@1='1440469563'

exec sp_executesql N'insert [Chapter10].[Book]([Title], [ISBN])values (@0, @1)

select [BookId]  from [Chapter10].[Book]

where @@ROWCOUNT > 0 and [BookId] = scope_identity()',N'@0 varchar(50),

@1 varchar(50)',@0='The Time Traveler''s Wife',@1='015602943X'

exec [Chapter10].[InsertAuthorBook] @AuthorId=1,@BookId=1

exec [Chapter10].[InsertAuthorBook] @AuthorId=1,@BookId=2

exec [Chapter10].[InsertAuthorBook] @AuthorId=2,@BookId=3

exec [Chapter10].[DeleteAuthorBook] @AuthorId=1,@BookId=1

exec sp_executesql N'delete [Chapter10].[Book] where ([BookId] = @0)',N'@0 int',@0=1

它是如何工作的?

为把存储过程映射到多对多关系中的插入和删除操作, 我们在数据库中创建存储过程,然后用存储过程模型.由于EF设计视图不支持关系型的模型映射插入和删除操作,我们需要用XML编辑器直接打开.edmx 文件, 在Mappings 节点里 <ModificationFunctionMapping> 标签下,我们添加了插入和删除操作映射到存储过程的代码。

从Listing 10-25跟踪生成的SQL,我们可以看到,不只是插入或删除 Author 和 Book 两个表, 同时我们也可以看到,存储过程在关系模型中插入和删除操作时被使用。

附:创建示例用到的数据库的脚本文件

Entity Framework 6 Recipes 2nd Edition(10-9)译 -> 在多对多关系中为插入和删除使用存储过程的更多相关文章

  1. Entity Framework 6 Recipes 2nd Edition 译 -> 目录 -持续更新

    因为看了<Entity Framework 6 Recipes 2nd Edition>这本书前面8章的翻译,感谢china_fucan. 从第九章开始,我是边看边译的,没有通读,加之英语 ...

  2. Entity Framework 6 Recipes 2nd Edition(9-1)译->用Web Api更新单独分离的实体

    第九章 在N层结构的应用程序中使用EF 不是所有的应用都能完全地写入到一个单个的过程中(就是驻留在一个单一的物理层中),实际上,在当今不断发展的网络世界,大量的应用程序的结构包含经典的表现层,应用程, ...

  3. Entity Framework 6 Recipes 2nd Edition(9-3)译->找出Web API中发生了什么变化

    9-3. 找出Web API中发生了什么变化 问题 想通过基于REST的Web API服务对数据库进行插入,删除和修改对象图,而不必为每个实体类编写单独的更新方法. 此外, 用EF6的Code Fri ...

  4. Entity Framework 6 Recipes 2nd Edition(9-4)译->Web API 的客户端实现修改跟踪

    9-4. Web API 的客户端实现修改跟踪 问题 我们想通过客户端更新实体类,调用基于REST的Web API 服务实现把一个对象图的插入.删除和修改等数据库操作.此外, 我们想通过EF6的Cod ...

  5. Entity Framework 6 Recipes 2nd Edition(13-4)译 -> 有效地创建一个搜索查询

    问题 你想用LINQ写一个搜索查询,能被转换成更有效率的SQL.另外,你想用EF的CodeFirst方式实现. 解决方案 假设你有如下Figure 13-6所示的模型 Figure 13-6. A s ...

  6. Entity Framework 6 Recipes 2nd Edition(13-2)译 -> 用实体键获取一个单独的实体

    问题 不管你用DBFirst,ModelFirst或是CodeFirst的方式,你想用实体键获取一个单独的实体.在本例中,我们用CodeFirst的方式. 解决方案 假设你有一个模型表示一个Paint ...

  7. Entity Framework 6 Recipes 2nd Edition(13-3)译 -> 为一个只读的访问获取实体

    问题 你想有效地获取只是用来显示不会更新的操作的实体.另外,你想用CodeFirst的方式来实现 解决方案 一个非常常见行为,尤其是网站,就是只是让用户浏览数据.大多数情况下,用户不会更新数据.在这种 ...

  8. Entity Framework 6 Recipes 2nd Edition(13-5)译 -> 使POCO的修改追踪更高

    问题 你正在使用POCO,你想提高修改跟踪的性能,同时使内存消耗更少.另外,你想通过EF的CodeFirst方式来实现. 解决方案 假设你有一个关于Account(帐户)和相关的Payments(支付 ...

  9. Entity Framework 6 Recipes 2nd Edition(13-9)译 -> 避免Include

    问题 你想不用Include()方法,立即加载一下相关的集合,并想通过EF的CodeFirst方式实现. 解决方案 假设你有一个如Figure 13-14所示的模型: Figure 13-14. A ...

随机推荐

  1. opencv在图像显示中文

    在图像定位和模式识别时,经常需要把结果标注到图片上,标注内容可以是数字字母.矩形框等(opencv支持的)或者是中文汉字(借助freetype). 1.显示数字/矩形框 #include <op ...

  2. 多线程爬坑之路-Thread和Runable源码解析之基本方法的运用实例

    前面的文章:多线程爬坑之路-学习多线程需要来了解哪些东西?(concurrent并发包的数据结构和线程池,Locks锁,Atomic原子类) 多线程爬坑之路-Thread和Runable源码解析 前面 ...

  3. django server之间通过remote user 相互调用

    首先,场景是这样的:存在两个django web应用,并且两个应用存在一定的联系.某些情况下彼此需要获取对方的数据. 但是我们的应用肯经都会有对应的鉴权机制.不会让人家随随便便就访问的对吧.好比上车要 ...

  4. angular实现统一的消息服务

    后台API返回的消息怎么显示更优雅,怎么处理才更简洁?看看这个效果怎么样? 自定义指令和服务实现 自定义指令和服务实现消息自动显示在页面的顶部,3秒之后消失 1. 显示消息 这种显示消息的方式是不是有 ...

  5. .Net Core MVC 网站开发(Ninesky) 2.3、项目架构调整(续)-使用配置文件动态注入

    上次实现了依赖注入,但是web项目必须要引用业务逻辑层和数据存储层的实现,项目解耦并不完全:另一方面,要同时注入业务逻辑层和数据访问层,注入的服务直接写在Startup中显得非常臃肿.理想的方式是,w ...

  6. 旺财速啃H5框架之Bootstrap(二)

    突然感觉不知道写啥子,脑子里面没水了,可能是因为今晚要出去浪,哈哈~~~提前提醒大家平安夜要回家哦,圣诞节生00000000000这么多蛋....继续 上一篇的已经把bootstrap了解个大概了,接 ...

  7. 深入node之Transform

    Transform流特性 在开发中直接接触Transform流的情况不是很多,往往是使用相对成熟的模块或者封装的API来完成流的处理,最为特殊的莫过于through2模块和gulp流操作.那么,Tra ...

  8. 【NLP】十分钟快览自然语言处理学习总结

    十分钟学习自然语言处理概述 作者:白宁超 2016年9月23日00:24:12 摘要:近来自然语言处理行业发展朝气蓬勃,市场应用广泛.笔者学习以来写了不少文章,文章深度层次不一,今天因为某种需要,将文 ...

  9. Struts2实现ajax的两种方式

    基于Struts2框架下实现Ajax有两种方式,第一种是原声的方式,另外一种是struts2自带的一个插件. js部分调用方式是一样的: JS代码: function testAjax() { var ...

  10. PHP之Memcache缓存详解

         Mem:memory缩写(内存):内存缓存 1.  断电或者重启服务器内存数据即消失,即临时数据: Memcache默认端口:11211 存入方式:key=>>value    ...