您的位置:首页 > 数据库

手把手教你:让EF动态支持新增表、动态支持多数据库

2016-07-25 10:16 441 查看
名词解释:此动态非运行时动态,让EF动态支持新增表、动态切换数据库意在不改变项目核心框架,

通过新增或者替换组件的方式达到标题目地。

 

一、先来点简单的,动态支持多数据库

AppDbContext实现:

<span class="kwrd">public</span> <span class="kwrd">class</span> AppDbContext:DbContext

{

<span class="kwrd">public</span> AppDbContext(<span class="kwrd">string</span> configKey)

: base(configKey)

{

 

}

 

<span class="kwrd">protected</span> <span class="kwrd">override</span> <span class="kwrd">void</span> OnModelCreating(DbModelBuilder modelBuilder)

{

<span class="kwrd">base</span>.OnModelCreating(modelBuilder);

}

}


在AppDbContext构造函数中添加configKey参数,通过configKey参数指定配置文件中的连接字符串的配置项;

 

创建IDbContextProvider接口,如下:

<span class="kwrd">public</span> <span class="kwrd">interface</span> IDbContextProvider

{

AppDbContext Get();

}


意图很明显了,通过IDbContextProvider来提供AppDbContext,这样我们首先将AppDbContext与业务层解耦;

 

继续创建2个项目:MsSqlProvider、MySqlProvider,分别实现IDbContextProvider接口:

MsSql:

<span class="kwrd">public</span> <span class="kwrd">class</span> MsSqlProvider:IDbContextProvider

{

AppDbContext m_AppDbContext = <span class="kwrd">null</span>;

 

<span class="kwrd">public</span> AppDbContext Get()

{

<span class="kwrd">return</span> m_AppDbContext ?? <span class="kwrd">new</span> AppDbContext(<span class="str">"MsSql"</span>);

}

}


MySQL:

<span class="kwrd">public</span> <span class="kwrd">class</span> MySqlProvider:IDbContextProvider

{

AppDbContext m_AppDbContext = <span class="kwrd">null</span>;

 

<span class="kwrd">public</span> AppDbContext Get()

{

<span class="kwrd">return</span> m_AppDbContext ?? <span class="kwrd">new</span> AppDbContext(<span class="str">"MySql"</span>);

}

}


下面继续解释动态支持/切换DbContextProvider,没错…聪明的你一开始就应该想到了..依赖注入,这个时候我们就需要使用依赖注入来完成使命了;

我已MEF为例来演示下如何动态获取2种DbContextProvider:

首先为我们的IDbContextProvider添加 [InheritedExport] 标记,然后分别为两种Provider添加 [Export]标记;

 

"MEF的使用还请大家自己去熟悉,我也仅仅是会使用而已,并不精通"

 

接着在Demo中添加App.Config和测试代码;

App.Config:

<?xml version=<span class="str">"1.0"</span> encoding=<span class="str">"utf-8"</span> ?>

<configuration>

<connectionStrings>

<add name="MsSql" connectionString="Data Source=LIANG-HU-PC;Initial Catalog=appbase;Integrated Security=True;Pooling=False" providerName="System.Data.SqlClient" />

<add name=<span class="str">"MySql"</span> connectionString=<span class="str">"server=localhost;User Id=root;password=mysql;Persist Security Info=True;database=appbase"</span> providerName=<span class="str">"MySql.Data.MySqlClient"</span> />

</connectionStrings>

</configuration>


这里要提醒下哦:要使MySql能够支持EF使用的话,需要到MySql官方下载最新的驱动;

测试代码如下:

<span class="kwrd">class</span> Program

{

[ImportMany]

static IEnumerable<IDbContextProvider> m_Providers = null;

 

static void Main(string[] args)

{

//使用目录方式查找MEF部件

var catalog = <span class="kwrd">new</span> DirectoryCatalog(AppDomain.CurrentDomain.BaseDirectory);

 

<span class="rem">//创建Container</span>

var container = new CompositionContainer(catalog);

 

//获取Export项,这里直接加载不采用Lazy

m_Providers = container.GetExportedValues<IDbContextProvider>();

if (m_Providers != null)

{

foreach (var provider in m_Providers)

{

Console.WriteLine(provider.Get().Database.Connection.ConnectionString);

}

}

Console.ReadKey(<span class="kwrd">false</span>);

}


OK,我们来编译测试下,当应用程序目录下没有任何Provider的时候是没有获取到任何是不会获取到任何Provider的,如果只放置MySqlProvider再执行的话结果如下:





放置两项Provider组件的时候自然就会是两个都被获取到,我就不演示了;

 

到这里可能很多人就要嘘声一片了,也许你会提出一些问题:

比如:

1)为什么不做一个Provider的实现,在Get()方法或者构造函数中依赖注入参数呢?

其实这样做的目地是我们在使用UnitOfWork和Repository模式时能够简单方便的获取DbContext;

可参见示例:

<span class="rem">/// <summary></span>

/// Entity Framework Repository

<span class="rem">/// </summary></span>

/// <typeparam name="T"></typeparam>

<span class="kwrd">public</span> <span class="kwrd">class</span> EFRepository<T>:IRepository<T>

where T:class

{

readonly IDataBaseFactory m_DataBaseFactory = null;

 

public EFRepository(IDataBaseFactory dataBaseFactory)

{

if(dataBaseFactory==null)

{

throw new ArgumentNullException("DataBaseFactory");

}

m_DataBaseFactory=dataBaseFactory;

}

 

DbContext m_DbContext = <span class="kwrd">null</span>;

 

<span class="kwrd">protected</span> DbContext DbContext

{

get

{

<span class="kwrd">return</span> m_DbContext ?? m_DataBaseFactory.Get();

}

}


对UnitOfWork模式的使用与此类似;

 

 


2)我只需要一个DbContext,但有时候需要切换数据库,那怎么办呢?

这个问题是ico与依赖注入方面的基础内容,需要您自己去学习哦;

 

至此,简单的“动态”支持多数据库示例就完成了~~~ 我们的关键还是动态支持新建表,下面我们就来一步一步实践吧;

 

二、“动态”支持新建表,计划先行

首先我们创建ModelBase类库,存放一些与实体相关的接口和基类,结构如图所示:



根据项目结构,我需要给大家解释每个文件的存在意义;

 

IEntity接口与AbstractEntityBase类,顾名思义,大家应该猜得到它们是实体基类,为什么要如此定义呢,主要是方便我们写实体的时候直接继承Id属性,(因为我们的所有表主键都是Guid且名为Id)

<span class="kwrd">public</span> <span class="kwrd">interface</span> IEntity

{

Guid Id { get; }

}

<span class="kwrd">public</span> <span class="kwrd">abstract</span> <span class="kwrd">class</span> AbstractEntityBase : IEntity

{

<span class="kwrd">public</span> AbstractEntityBase()

{

<span class="kwrd">this</span>.Id = Guid.NewGuid();

}

 

[Key]

[Required]

public Guid Id

{

get;

<span class="kwrd">protected</span> set;

}

}


还有一个好处就是我们直接在基类中描述 主键关系,在写实体的时候直接继承后,可以省去很多重复操作哦^_^

 

再来说IMapping和Mapping,为什么要有这2个基类接口呢,出于以下方面考虑:

1)将实体与数据库的映射关系产生Mapping类与DbContext类解耦(这个会在下面具体出现时再说)

2)通过MappingBase基类实现一些公共操作,避免每个实体类的重复操作,具体看代码你就会明白;

[InheritedExport]

public interface IMapping

{

void RegistTo(ConfigurationRegistrar configurationRegistrar);

}

public class MappingBase<TEntity> : EntityTypeConfiguration<TEntity>, IMapping

<span class="kwrd">where</span> TEntity : <span class="kwrd">class</span>,IEntity

{

<span class="kwrd">public</span> MappingBase()

{

<span class="kwrd">this</span>.Map(m => m.ToTable(<span class="kwrd">typeof</span>(TEntity).Name));

}

 

public void RegistTo(ConfigurationRegistrar configurationRegistrar)

{

configurationRegistrar.Add(this);

}

}


呵呵,有了“动态”支持多数据库,这里很多人应该就能猜到我们如何“动态”支持新增表咯;注意这里的IMapping接口的精妙所在哦,您发现了吗???;

 

三、万事俱备,只欠东风

我们先在ModelA类库中创建一个User实体和Role实体,同时创建UserMapping和RoleMapping,(为什么要创建Mapping类,后面我会讲)

USer 、UserMapping:

<span class="rem">/*</span>

* 为什么没有通过来指明表明呢,
<span class="rem">     * 并不是因为我们需要EF自己支持的表明方式</span>

* 而是我们继承自AbstractEntityBase,在其基类已经实现了将类名映射为表名

<span class="rem">*/</span>

public class User : AbstractEntityBase

{

[Required]

<span class="kwrd">public</span> <span class="kwrd">string</span> Username { get; set; }

 

[Required]

public string Password { get; set; }

 

/*

<span class="rem">         * 注意这里,我为什么不通过DataAnnotations方式添加外键关联呢</span>

* 个人认为User实体与Role实体关联,已经拥有Role属性了,

<span class="rem">         * 如果在添加一个RoleId来表示外键关系,会让我觉得User类不够清爽</span>

* 所以我的做法是添加UserMapping类来指定它与Role实体的关系

<span class="rem">         * </span>

* 但是有一点要注意,如果不指定外键的话,默认数据库外键是为 表名_主键(Role_Id)类型

<span class="rem">    */</span>

public virtual Role Role{get;set;}

}

 

[Export(<span class="str">"UserMapping"</span>)]

public class UserMapping:MappingBase<User>

{

public UserMapping()

{

this.HasRequired(m => m.Role)

.WithMany(m => m.Users);

    /*注意这里没有指定HasForeignKey哦*/

}

}


Role类和RoleMapping的实现也是同理,结合上面代码中的注释内容,我想大家也能够理解我的良苦用心了吧;如果还不能理解,我们再看下DbContext是如何实现的:

<span class="rem">/*</span>

* 很清爽的DbContext,完全不包含任何DbSet

<span class="rem">     * 通过Mapping来加载表结构</span>

*/

<span class="kwrd">public</span> <span class="kwrd">class</span> AppDbContext:DbContext

{

<span class="kwrd">public</span> AppDbContext(<span class="kwrd">string</span> configKey)

: base(configKey)

{

//可以设置通过反向方式创建表哦,但是我们演示的目地不在于此

<span class="rem">//Database.SetInitializer(new DropCreateDatabaseIfModelChanges<AppDbContext>());</span>

 

<span class="rem">//加载目录下所有IMapping实现</span>

var catalog = new DirectoryCatalog(AppDomain.CurrentDomain.BaseDirectory);

var container = <span class="kwrd">new</span> CompositionContainer(catalog);

m_Mappings = container.GetExportedValues<IMapping>();

}

 

[ImportMany]

IEnumerable<IMapping> m_Mappings = null;

 

protected override void OnModelCreating(DbModelBuilder modelBuilder)

{

if (m_Mappings != null)

{

//这里是关键

<span class="kwrd">foreach</span> (var mapping <span class="kwrd">in</span> m_Mappings)

{

mapping.RegistTo(modelBuilder.Configurations);

}

}

base.OnModelCreating(modelBuilder);

}

}


 

没错,我们的目地就是让DbContext完全依靠IMapping去加载解释表结构关系,这样即保证DbContext不包含大量的DbSet,又能够非常好的将DbContext与实体解耦,

更重要的是,我们通过 DataAnnotations 和 Fluent API结合使用,让我们的实体也非常清爽;

 

到这里,其实我已经把核心的内容都展现出来了,对于新增表的动态使用也就类似与最前面讲的“动态”支持多数据库,我们只需要依赖注入所有的IMapping的实现,就可以让DbContext自动去解释所有表结构了(所以DbContext的OnModelCreating方法是关键所在)。

 

好,接着我们在新增一个ModelB 作为新增表NewModel实体的载体,来演示是我们的示例是否能够如题所描述的那样,不改变核心框架的前提下动态支持新增的表和实体。

<span class="kwrd">public</span> <span class="kwrd">class</span> NewModel : AbstractEntityBase

{

[Required]

public string Name { get; set; }

}


 

我们按照之前做CmdDemo的方式添加一个AppDemo,并添加App.Config文件,同时创建一个DataViewControl的自定义控件用来显示数据;

我们来看下AppDemo的演示:





其具体实现为:

<span class="rem">/// <summary></span>

/// MainWindow.xaml 的交互逻辑

<span class="rem">/// </summary></span>

public partial class MainWindow : Window

{

public MainWindow()

{

InitializeComponent();

}

 

<span class="kwrd">private</span> <span class="kwrd">void</span> Window_Loaded(<span class="kwrd">object</span> sender, RoutedEventArgs e)

{

IDbContextProvider provider = <span class="kwrd">new</span> MsSqlProvider.MsSqlProvider();

AppDbContext dbContext = provider.Get();

 

var users = dbContext.Set<User>();

<span class="kwrd">if</span> (users != <span class="kwrd">null</span>)

{

users.Add(<span class="kwrd">new</span> User()

{

Username = <span class="str">"admin"</span>,

Password = "admin",

Role = <span class="kwrd">new</span> Role() { Name = <span class="str">"administrators"</span> }

});

dbContext.SaveChanges();

 

DataViewControl usersViewControl=<span class="kwrd">new</span> DataViewControl();

usersViewControl.Binding(users.ToList());

 

TabItem item = new TabItem();

item.Header = <span class="str">"User表展示"</span>;

item.Content = usersViewControl;

 

this.myTabControl.Items.Add(item);

}

 

var roles = dbContext.Set<Role>();

if (roles != null)

{

DataViewControl rolesViewControl = new DataViewControl();

rolesViewControl.Binding(roles.ToList());

 

TabItem item = <span class="kwrd">new</span> TabItem();

item.Header = "Role表展示";

item.Content = rolesViewControl;

this.myTabControl.Items.Add(item);

}

 

           <span class="rem">/*</span>

* 请注意此处,我们的NewModel还是和应用耦合在一起了,

<span class="rem">             * 并没有像我们标题说的动态加载;</span>

* 这里主要是为了演示方便,我就不在做实体与业务层的解耦了,

<span class="rem">             * 一般我们的应用可能是单独的UI模块和它对应的实体耦合,而不是UI框架耦合</span>

* 仅在需要的时候加载不同模块的UI组件

<span class="rem">             *</span>

        */

var newModels = dbContext.Set<NewModel>();

if (newModels != null)

{

DataViewControl newModelsViewControl = new DataViewControl();

newModelsViewControl.Binding(newModels.ToList());

TabItem item = new TabItem();

item.Header = <span class="str">"NewModel表展示"</span>;

item.Content = newModelsViewControl;

<span class="kwrd">this</span>.myTabControl.Items.Add(item);

}

}

}


 

需要解释的是AppDemo中没有很好的演示怎么动态支持新建表,其实我前面解释过ModelB中NewModel就是新增的表,主要是为了给大家展示实现思路,我并没有去把NewModel和AppDemo去解耦,所以没有很好的演示效果,但是实际上是没有问题的,这就跟我们具体的应用息息相关了。

到这里,我们已经完整的解释了整个过程,为此我也是边创建项目边写博客,在最后会附上完整项目源码,有兴趣的可以自行下载学习;如果这篇文章对你有所启发,或者让你学到了一些东西,那是我非常乐见的,同时也希望各位高手不要鄙视。
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: