问题:
在实体类型上生成一个永久性的过滤,使其能够映射为表中记录的子集。
解决方案:
有一张保存账号信息的表,如下图所示。这个表有一个DeletedOn的可空列,用于保存账号被删除的日期和时间。如果账号仍然是激活的,这个列为空。我们想我们的Account集合仅仅表示激活的账号,也就是这个账号没有DeletedOn值。
数据库脚本如下:
create table [Chapter2].[Account] ( AccountID int primary key identity(1,1), DeletedOn datetime null, AccountHolderID int not null )
为了建模这张表,使其仅有激活账号填充Account实体集合,完成下面的步骤:
1、添加一个新的EDM模型。
2、选择来自数据库的EF设计器。
3、选择数据连接。
4、在数据库对象和设置对话框中,选择Account表,并保持默认设置。点击“完成”。点击“EF设计器”,选择属性,设置“实体容器名”为“EF6Recipes9Context”;并修改名称空间。
5、在EF设计器中,右击Account实体,选择“表映射”,打开“映射细节”窗口。点击“添加条件”,选择DeletedOn列;在操作列,选择“是”;在值/属性列,选择“Null”。这样就产生了一个映射条件:当DeletedOn为空时。
6、在EF设计器中,选中Account实体的DeletedOn属性,右击,选择“删除”。由于我们正在条件映射中使用DeletedOn列,所以它不能被映射到实体的一个属性。在我们的模型中DeletedOn始终为Null。
原理:
当我们想在一个实体上应用永久性过滤时,我们常常使用条件映射。条件映射也是实现TPH继承的关键。可应用的条件有:
<value> Is Null
<value> Is Not Null
<integer> = <value>
<string> = <value>
using (var context = new EF6Recipes9Context()) { context.Database.ExecuteSqlCommand(@"insert into chapter2.account(DeletedOn,AccountHolderId) values('2/10/2009',1728)"); var account = new Account { AccountHolderID = 2320 }; context.Accounts.Add(account); account = new Account { AccountHolderID = 2502 }; context.Accounts.Add(account); account = new Account { AccountHolderID = 2603 }; context.Accounts.Add(account); context.SaveChanges(); } using (var context = new EF6Recipes9Context()) { foreach (var account in context.Accounts) { Console.WriteLine("Account Id = {0}", account.AccountHolderID); } }
上面的代码中,我们首先使用Context的Database属性的ExecuteSqlCommand方法向数据库中插入一条记录。主要是因为我们需要插入一条DeletedOn列为非空值的情况。在我们的模型中没有这样的属性映射的这个列。事实上,Account实体类型从不实例化一个DeletedOn为非空的实例。
上面的结果并不包含我们最先插入的DeletedOn为非空的记录。
免责声明:本站发布的内容(图片、视频和文字)以原创、转载和分享为主,文章观点不代表本网站立场,如果涉及侵权请联系站长邮箱:is@yisu.com进行举报,并提供相关证据,一经查实,将立刻删除涉嫌侵权内容。