我一直在尝试使用EntityFramework5实现Repository模式。我已经掌握了基础知识。我的存储库仅在必要时加载,数据正在数据库中插入/更新/检索。 问题是:我无法获得正确创建映射的表,甚至根本不创建。 我一直在网上阅读一些文章,甚至在这里,谈论使用映射表并在数据库中创建它们的方法,但我没有成功。在这里提出我的问题。
如何让它在我的场景中运行?
我会解释到目前为止我在做什么:
下面的方法被覆盖,据说可以创建映射和表。
PS:Generics.InvokeGenericMethod
是我创建的一种方法,它正在运行。它做了它所说的:)
protected override void OnModelCreating(DbModelBuilder modelBuilder)
{
foreach (Type entityType in Repositories.Keys)
{
//Invokes the Entity method from DbModelBuilder injecting the class type.
Generics.InvokeGenericMethod(modelBuilder, typeof(DbModelBuilder), entityType, "Entity", null);
}
foreach (GaiaEntityConfiguration config in EntityConfigurations)
{
//Defines each mapping existing in the context
config.SetConfiguration(modelBuilder);
}
}
无效的部分是我使用SetConfiguration的地方。 SetConfiguration是我为添加到具有映射的上下文的每个类创建的方法,将该映射镜像到数据库。
这是一个例子(在这种情况下,收件人有很多邮件,邮件有很多收件人/多对多关联):
PS:我留下了注释代码,以便您可以看到我尝试的另一种方法。
public RecipientEntityConfiguration()
{
this.LeftKey = "RecipientId";
this.RightKey = "MessageId";
this.Schema = "Hermes";
this.TableName = "RecipientMessage";
}
public override void SetConfiguration(DbModelBuilder modelBuilder)
{
//EntityTypeConfiguration<Recipient> entityTypeConfiguration = new EntityTypeConfiguration<Recipient>();
//entityTypeConfiguration
// .HasMany<Message>(r => r.Messages)
// .WithMany(m => m.Recipients)
// .Map(mr =>
// {
// mr.MapLeftKey(this.LeftKey);
// mr.MapRightKey(this.RightKey);
// mr.ToTable(this.TableName, this.Schema);
// });
modelBuilder.Entity<Recipient>()
.HasMany<Message>(r => r.Messages)
.WithMany(m => m.Recipients)
.Map(mr =>
{
mr.MapLeftKey(this.LeftKey);
mr.MapRightKey(this.RightKey);
mr.ToTable(this.TableName, this.Schema);
});
//modelBuilder.Configurations.Add<Recipient>(entityTypeConfiguration);
}
调用this.Database.Initialize(false);
时出现此错误:
Value cannot be null.
Parameter name: key
堆栈跟踪:
at System.Collections.Generic.Dictionary`2.FindEntry(TKey key)
at System.Collections.Generic.Dictionary`2.TryGetValue(TKey key, TValue& value)
at System.Data.Entity.ModelConfiguration.Configuration.Mapping.SortedEntityTypeIndex.Add(EdmEntitySet entitySet, EdmEntityType entityType)
at System.Data.Entity.ModelConfiguration.Configuration.Mapping.EntityMappingService.Analyze()
at System.Data.Entity.ModelConfiguration.Configuration.Mapping.EntityMappingService.Configure()
at System.Data.Entity.ModelConfiguration.Configuration.ModelConfiguration.ConfigureEntityTypes(DbDatabaseMapping databaseMapping, DbProviderManifest providerManifest)
at System.Data.Entity.ModelConfiguration.Configuration.ModelConfiguration.Configure(DbDatabaseMapping databaseMapping, DbProviderManifest providerManifest)
at System.Data.Entity.DbModelBuilder.Build(DbProviderManifest providerManifest, DbProviderInfo providerInfo)
at System.Data.Entity.DbModelBuilder.Build(DbConnection providerConnection)
at System.Data.Entity.Internal.LazyInternalContext.CreateModel(LazyInternalContext internalContext)
at System.Data.Entity.Internal.RetryLazy`2.GetValue(TInput input)
at System.Data.Entity.Internal.LazyInternalContext.InitializeContext()
at System.Data.Entity.Internal.InternalContext.Initialize()
at System.Data.Entity.Database.Initialize(Boolean force)
at Gaia.Repository.GaiaContext..ctor(GaiaContextConfiguration config) in E:\Gaia\Gaia.Repository\GaiaContext.cs:line 37
at Hermes.HMail.SendMessage(Int64 sender, Int64[] toUsers, String subject, String content, FileStream attachment) in G:\Gaia\Hermes\HMail.cs:line 78
at Gaia.Controllers.ApplicationController.Test() in G:\Gaia\Gaia\Controllers\ApplicationController.cs:line 18
at lambda_method(Closure , ControllerBase , Object[] )
at System.Web.Mvc.ActionMethodDispatcher.<>c__DisplayClass1.<WrapVoidAction>b__0(ControllerBase controller, Object[] parameters)
at System.Web.Mvc.ActionMethodDispatcher.Execute(ControllerBase controller, Object[] parameters)
at System.Web.Mvc.ReflectedActionDescriptor.Execute(ControllerContext controllerContext, IDictionary`2 parameters)
at System.Web.Mvc.ControllerActionInvoker.InvokeActionMethod(ControllerContext controllerContext, ActionDescriptor actionDescriptor, IDictionary`2 parameters)
at System.Web.Mvc.ControllerActionInvoker.<>c__DisplayClass15.<InvokeActionMethodWithFilters>b__12()
at System.Web.Mvc.ControllerActionInvoker.InvokeActionMethodFilter(IActionFilter filter, ActionExecutingContext preContext, Func`1 continuation)
有什么想法吗?
感谢。
答案 0 :(得分:2)
首先,我在EF Book pg.57中发现,任何内联配置都应该在所有配置添加到modelbuilder之后出现。检查你的代码。
接下来,尝试使用此版本的代码,以便自动将映射添加到modelbuilder。 目标是从EntityTypeConfigurations中删除调用,并使用向上下文添加配置的纯方法。将代码从SetConfiguration移动到构造函数并尝试使用我的版本。它对我也很有用。
protected override void OnModelCreating(DbModelBuilder modelBuilder)
{
var entityMapTypes = assembly.GetTypes().Where(
(t => t.BaseType != null && t.BaseType.IsGenericType &&
t.BaseType.GetGenericTypeDefinition() == typeof(EntityTypeConfiguration<>)));
foreach (var mapType in entityMapTypes)
{
dynamic configurationInstance = Activator.CreateInstance(mapType);
modelBuilder.Configurations.Add(configurationInstance);
}
}
答案 1 :(得分:1)
这将是我在stackoverflow中的第一个答案,所以我希望它可以帮助某人!原来EF是一个敏感的生物,它不会欣赏具有名为“Type”属性的实体,所以如果你有任何实体,只需将它们重命名为“EntityType”。
我也听说过人们对具有抽象类型的属性有类似问题,以防有人的问题不是我上面描述的问题,看看你的属性的类型并检查它们是否是抽象的: DbContext fails to initialize Model-first database