我正在尝试使用Entity Framework的代码优先概念创建一个新的数据库。但是,在运行代码时,不会创建数据库(使用DropCreateDatabaseIfModelChanges
设置),尽管代码运行得很好。当我试图从数据库中获取一些东西时,我看到了下面的异常。
我的项目是使用具有通用服务和存储库构造的单独DataAccess
层进行设置的。因此,我的所有实体、存储库和数据库上下文都在解决方案中的一个单独项目中。
我的global.asax
文件包含以下代码段。
Database.SetInitializer(new DropCreateDatabaseIfModelChanges<MyContext>());
这应该会初始化一个新的数据库,如果它不在那里,对吗?
我的数据库上下文类如下所示;
namespace Website.DAL.Model
{
public class MyContext : DbContext
{
public IDbSet<Project> Projects { get; set; }
public IDbSet<Portfolio> Portfolios { get; set; }
/// <summary>
/// The constructor, we provide the connectionstring to be used to it's base class.
/// </summary>
public MyContext()
: base("MyConnectionString")
{
}
static MyContext()
{
try
{
Database.SetInitializer<MyContext>(new DropCreateDatabaseIfModelChanges<MyContext>());
}
catch (Exception)
{
throw;
}
}
/// <summary>
/// This method prevents the plurarization of table names
/// </summary>
/// <param name="modelBuilder"></param>
protected override void OnModelCreating(DbModelBuilder modelBuilder)
{
base.OnModelCreating(modelBuilder);
modelBuilder.Conventions.Remove<System.Data.Entity.ModelConfiguration.Conventions.PluralizingTableNameConvention>();
}
}
}
我根据互联网上的一些教程和文章创建了这个类。这对我来说是全新的,但据我所知,到目前为止一切似乎都是正确的。现在我使用的两个实体。它们被称为'Project‘和'Portfolio’。它们看起来像这样;
public class Portfolio
{
[Key]
public Guid Id { get; set; }
public String Name { get; set; }
public DateTime StartDate { get; set; }
public DateTime? EndDate { get; set; }
public bool IsPublished { get; set; }
public virtual ICollection<Project> Projects { get; set; }
}
和
public class Project
{
[Key]
public Guid Id { get; set; }
public DateTime StartDate { get; set; }
public DateTime? EndDate { get; set; }
public bool IsPublished { get; set; }
public String Title { get; set; }
}
我使用的数据库是在外部服务器上运行的,它与我使用的主机提供商一起提供。我已经启动并运行了一个SQL Server数据库,该数据库的连接字符串位于网站项目的web.config
中。我已经尝试删除数据库,并让代码重新创建它,不幸的是,这不起作用。我是不是漏掉了什么明显的东西?或者这可能是一件简单的事情,比如创建数据库的服务器的访问权限?
注意:当我运行Database-Update -Script
命令来生成SQL代码时,似乎创建了用于创建所有表的正确SQL语句。
更新1:好的,多亏了一些评论,我走得更远了。我已经向实体添加了两个属性来强制进行一些更改,并且我还创建了一个自定义初始化器,如下所示;
public class ForceDeleteInitializer : IDatabaseInitializer<MyContext>
{
private readonly IDatabaseInitializer<MyContext> _initializer = new DropCreateDatabaseIfModelChanges<MyContext>();
public ForceDeleteInitializer()
{
//_initializer = new ForceDeleteInitializer();
}
public void InitializeDatabase(MyContext context)
{
//This command is added to prevent open connections. See http://stackoverflow.com/questions/5288996/database-in-use-error-with-entity-framework-4-code-first
context.Database.ExecuteSqlCommand("ALTER DATABASE borloOntwikkel SET SINGLE_USER WITH ROLLBACK IMMEDIATE");
_initializer.InitializeDatabase(context);
}
}
我还从上下文的构造函数中删除了初始化器,所以这意味着我删除了这行代码;
Database.SetInitializer<MyContext>(new DropCreateDatabaseIfModelChanges<MyContext>());
之后,我将这三行添加到我的Global.asax文件中;
Database.SetInitializer(new ForceDeleteInitializer());
MyContext c = new MyContext();
c.Database.Initialize(true);
在调试时,我现在得到了这个异常;
这为我提供了以下信息:
执行这些操作后,数据库将无法访问,因此很可能已删除。
对此我们能做些什么呢?我很可能无法访问master数据库,因为我的主机提供商当然不会给我适当的访问权限。
发布于 2013-07-30 18:42:05
由于没有其他的解决方案,我决定改变我的方法。
我首先自己创建了数据库,并确保配置了正确的SQL用户,并且我拥有访问权限。
然后,我从Global.asax文件中删除了初始化器和代码。之后,我在Package Manager控制台中运行以下命令(因为分层设计,我必须在控制台中选择正确的项目);
Enable-Migrations
在启用迁移之后,我对我的实体做了一些最后的修改,我运行下面的命令来搭建一个新的迁移;
Add-Migration AddSortOrder
创建迁移后,我在控制台中运行以下命令,瞧,数据库已经用我的实体更新了;
Update-Database -Verbose
为了能够在运行迁移时为数据库设定种子,我覆盖了Configuraton.cs类中的种子方法,该类是在启用迁移时创建的。此方法的最终代码如下所示;
protected override void Seed(MyContext context)
{
// This method will be called after migrating to the latest version.
//Add menu items and pages
if (!context.Menu.Any() && !context.Page.Any())
{
context.Menu.AddOrUpdate(
new Menu()
{
Id = Guid.NewGuid(),
Name = "MainMenu",
Description = "Some menu",
IsDeleted = false,
IsPublished = true,
PublishStart = DateTime.Now,
LastModified = DateTime.Now,
PublishEnd = null,
MenuItems = new List<MenuItem>()
{
new MenuItem()
{
Id = Guid.NewGuid(),
IsDeleted = false,
IsPublished = true,
PublishStart = DateTime.Now,
LastModified = DateTime.Now,
PublishEnd = null,
Name = "Some menuitem",
Page = new Page()
{
Id = Guid.NewGuid(),
ActionName = "Some Action",
ControllerName = "SomeController",
IsPublished = true,
IsDeleted = false,
PublishStart = DateTime.Now,
LastModified = DateTime.Now,
PublishEnd = null,
Title = "Some Page"
}
},
new MenuItem()
{
Id = Guid.NewGuid(),
IsDeleted = false,
IsPublished = true,
PublishStart = DateTime.Now,
LastModified = DateTime.Now,
PublishEnd = null,
Name = "Some MenuItem",
Page = new Page()
{
Id = Guid.NewGuid(),
ActionName = "Some Action",
ControllerName = "SomeController",
IsPublished = true,
IsDeleted = false,
PublishStart = DateTime.Now,
LastModified = DateTime.Now,
PublishEnd = null,
Title = "Some Page"
}
}
}
});
}
if (!context.ComponentType.Any())
{
context.ComponentType.AddOrUpdate(new ComponentType()
{
Id = Guid.NewGuid(),
IsDeleted = false,
IsPublished = true,
LastModified = DateTime.Now,
Name = "MyComponent",
PublishEnd = null,
PublishStart = DateTime.Now
});
}
try
{
// Your code...
// Could also be before try if you know the exception occurs in SaveChanges
context.SaveChanges();
}
catch (DbEntityValidationException e)
{
//foreach (var eve in e.EntityValidationErrors)
//{
// Console.WriteLine("Entity of type \"{0}\" in state \"{1}\" has the following validation errors:",
// eve.Entry.Entity.GetType().Name, eve.Entry.State);
// foreach (var ve in eve.ValidationErrors)
// {
// Console.WriteLine("- Property: \"{0}\", Error: \"{1}\"",
// ve.PropertyName, ve.ErrorMessage);
// }
//}
//throw;
var outputLines = new List<string>();
foreach (var eve in e.EntityValidationErrors)
{
outputLines.Add(string.Format(
"{0}: Entity of type \"{1}\" in state \"{2}\" has the following validation errors:",
DateTime.Now, eve.Entry.Entity.GetType().Name, eve.Entry.State));
foreach (var ve in eve.ValidationErrors)
{
outputLines.Add(string.Format(
"- Property: \"{0}\", Error: \"{1}\"",
ve.PropertyName, ve.ErrorMessage));
}
}
System.IO.File.AppendAllLines(@"c:\temp\errors.txt", outputLines);
throw;
}
}
目前的缺点是,我必须(仅)使用包管理器控制台中的两个命令手动迁移。但同时,这不是动态发生的事实也是好的,因为这可以防止对我的数据库进行可能不必要的更改。更进一步说,一切都很完美。
发布于 2013-07-12 03:33:19
详细问题为+1。
检查连接字符串是否指向正确的数据库,并添加如下授权属性以访问数据库:
<add name="PatientContext" providerName="System.Data.SqlClient" connectionString="Server=SQLSERVER2; Database=Patients; uid=PatientUser; password=123456; Integrated Security=False;" />
发布于 2017-09-30 16:01:27
在初始化它之前,运行以下代码来创建数据库:
context.Database.CreateIfNotExists();
https://stackoverflow.com/questions/17601459
复制相似问题