有一段时间没有更新博客了,赶上今天外面下雨,而且没人约球,打算把最近对entity framework dbcontext使用的心得梳理一下,早些时候在网上简单查过,对于最新版本的ef并没有类似的知识梳理类文章,希望对大家有所帮助。
1. 不要code first, 也不要db first
我为什么讨厌code first和db first呢?首先code first是先写代码,数据库完全由代码生成,开发阶段尚可,一旦到了产品发布阶段,如果需要添加字段,我们总不能用 visual studio去生产环境上去更新数据库吧,听起来就很可怕。而且另外的一个问题自动是生成的数据库脚本也不可控,还不如自己提前设计好。db first也好不了哪去,反向转过来的代码包含很多没有用的文件,而且数据库的更新还要重新走model生成过程,简直无法理解为什么会有这样的设计。说了这么多,怎么解决呢?
数据库和领域模型分开设计,按照对应关系映射字段,使用自定义链接字串,既不使用领域模型生成数据库,也不用数据库生成领域模型,示例代码如下,sql code 以 destinations和ttable表为例:
create table [dbo].[destinations]
(
[destinationid] [int] primary key not null,
[name] [nvarchar](max) null,
[country] [nvarchar](max) null,
[description] [nvarchar](max) null,
[photo] [varbinary](max) null
create table [ttt].[ttable] ( [id] [int] primary key not null, [name] [nvarchar](max) null )
model class:
using system;
using system.collections.generic;
using system.linq;
using system.text;
using system.threading.tasks;
namespace model
{
public class destination
{
public int destinationid { get; set; }
public string name { get; set; }
public string country { get; set; }
public string description { get; set; }
public byte[] photo { get; set; }
public list<lodging> lodgings { get; set; }
}
public class lodging
{
public int lodgingid { get; set; }
public string name { get; set; }
public string owner { get; set; }
public bool isresort { get; set; }
public destination destination { get; set; }
}
public class ttable
{
public int id { get; set; }
public string name { get; set; }
}
}connect string:
<connectionstrings>
<add name="bacontext" connectionstring="data source=(localdb)\mssqllocaldb;initial catalog=dataaccess.breakawaycontext;integrated security=sspi;" providername="system.data.sqlclient" />
</connectionstrings>db context:
using system.data.entity;
using system.data.entity.modelconfiguration;
using model;
namespace dataaccess
{
public class ttableconfiguration : entitytypeconfiguration<ttable>
{
public ttableconfiguration()
{
this.totable("ttable", "ttt");
}
}
public class breakawaycontext : dbcontext
{
protected override void onmodelcreating(dbmodelbuilder modelbuilder)
{
modelbuilder.configurations.add(new ttableconfiguration());
}
public breakawaycontext(string connstring) : base(connstring)
{
}
public dbset<destination> destinations { get; set; }
public dbset<lodging> lodgings { get; set; }
public dbset<ttable> ttables { get; set; }
}
}2. 如果数据库的表的字段和领域模型的字段不对应,如何处理呢?比如本文的ttable表是在ttt schema下面的, 而其他表示设计在dbo下面,最方便的方式是使用fluent api, 具体代码如请参见 ttableconfiguration class和 onmodelcreating()方法,可配置的粒度非常细,比如可以配置领域模型和数据库的哪个schema的哪张表的哪一列对应,本文是将ttable 类的数据库表配置为了ttt schema下的ttable表,
public class ttableconfiguration : entitytypeconfiguration<ttable>
{
public ttableconfiguration()
{
this.totable("ttable", "ttt");
}
}3. 增删该查自带事物支持,具体代码如下,
public static int insert()
{
var destination = new destination
{
country = "chs",
description = "chs is the language package",
name = "xsss"
};
using (var context = new breakawaycontext(configurationmanager.connectionstrings["bacontext"].connectionstring))
{
var rt = context.destinations.add(destination);
context.savechanges();
return rt.destinationid;
}
}
public static void update(destination destin)
{
using (var context = new breakawaycontext(configurationmanager.connectionstrings["bacontext"].connectionstring))
{
var dest = context.destinations.where(a => a.destinationid == destin.destinationid).single();
dest.name = destin.name;
context.savechanges();
}
}
public static void delete(int destid)
{
using (var context = new breakawaycontext(configurationmanager.connectionstrings["bacontext"].connectionstring))
{
var destination = new destination() { destinationid = destid };
context.destinations.attach(destination);
context.destinations.remove(destination);
context.savechanges();
}
}
public static destination query(int destid)
{
using (var context = new breakawaycontext(configurationmanager.connectionstrings["bacontext"].connectionstring))
{
iqueryable<destination> dest = context.destinations.where(a => a.destinationid == destid);
return dest.single();
}
}4. 如果需要多个操作同时成功或者失败,需要手动开启事务,具体代码如下,
public static void transactionops()
{
using (var context = new breakawaycontext(configurationmanager.connectionstrings["bacontext"].connectionstring))
{
using (var dbcontexttransaction = context.database.begintransaction())
{
try
{
var destination = new destination
{
country = "chs",
description = "chs is the language package",
name = "xs2s"
};
var destid = context.destinations.add(destination);
context.savechanges();
context.destinations.attach(destid);
context.destinations.remove(destid);
context.savechanges();
dbcontexttransaction.commit();
}
catch (system.exception ex)
{
dbcontexttransaction.rollback();
system.console.writeline(ex.tostring());
}
}
}
}5. 分页查询是网站设计的常用功能,一个简单的真分页查询方法如下如下所示,
public static list<destination> querypaging<tkey>(int pageindex, int pagesize, expression<func<destination, bool>> wherelambda, expression<func<destination, tkey>> orderby)
{
using (var context = new breakawaycontext(configurationmanager.connectionstrings["bacontext"].connectionstring))
{
return context.destinations.where(wherelambda).orderby(orderby).skip((pageindex - 1) * pagesize).take(pagesize).tolist();
}
}总结
本文对最新版本的entity framework进行增删改查操作给出了详尽的解释,并且给出了数据库和领域模型代码分开设计的完整解决方案,同时介绍了手动数据库表和领域模型映射,数据库事务实现,分页查询等常用功能,希望对大家有所帮助。
以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,谢谢大家对代码网的支持。如果你想了解更多相关内容请查看下面相关链接
发表评论