Entity Framework 代码首先导致数据库访问异常

Entity Framework code first causes exception on database access

最近开始尝试深入研究 Entity Framework 的代码优先方法。我按照各种教程创建了一些数据库表,但 none 成功了。

现在从头开始大约 3 到 4 次,因为异常变得越来越荒谬。上次在第一次访问时创建了数据库,但由于缺少权限(使用集成安全性 - wat?)而无法在第二次调试会话中连接到数据库。显然 .mdf 被系统进程阻止,只有在安全模式下重新启动才能让我摆脱它。

无论如何,重新来过。我有一个 class 库,其中包含数据库模型和一个用于外部访问的网关 class。我有一个控制台项目来测试数据库的创建和访问。 EF 通过 NuGet 安装在 class 库中,对 EntityFramework.SqlServer.dll 的引用被添加到控制台项目中。

两个项目中的连接字符串如下所示:

<connectionStrings>
    <add name="DbConnection"
        connectionString="Data Source=(LocalDb)\v11.0;AttachDbFilename=|DataDirectory|\LanguageCreator.mdf;Initial Catalog=LanguageCreator;Integrated Security=SSPI"
        providerName="System.Data.SqlClient" />
</connectionStrings>

我的上下文 class 看起来像这样:

public class LanguageContext : DbContext {
    public LanguageContext() : base("DbConnection") {
        // Have to set the DataDirectory in code because the wrong one gets set from the config.
        AppDomain.CurrentDomain.SetData("DataDirectory", Directory.GetCurrentDirectory());

        Database.SetInitializer(new CreateDatabaseIfNotExists<LanguageContext>());

        public DbSet<Language> Languages { get; set; }
        public DbSet< ... blablabla various other DbSets following, nothing of interest here.
    }

这是我的网关的样子:

public class Gateway {
    public void InitializeDatabase() {
        using (LanguageContext context = new LanguageContext()) {
            context.Database.Initialize(true);

            // I did have a configuration class for migration, which seeds some initial data, which is why I'm trying to read the WordTypes here but I left that thing out this time ... will try to add it later on.
            IEnumerable<WordType> wordTypes = context.WordTypes.AsEnumerable();

            List<Language> languages = context.Languages.ToList();
        }
    }
}

调用看起来像这样:

class Program {
    static void Main(string[] args) {
        Gateway databaseGateway = new Gateway();

        databaseGateway.InitializeDatabase();
    }
}

现在,没有什么特别的,我在启动调试会话时收到以下错误:

A network-related or instance-specific error occurred while establishing a connection to SQL Server. The server was not found or was not accessible. Verify that the instance name is correct and that SQL Server is configured to allow remote connections. (provider: SQL Network Interfaces, error: 50 - Local Database Runtime error occurred. Cannot create an automatic instance. See the Windows Application event log for error details.

不幸的是,Windows 应用程序事件日志没有显示该错误的任何详细信息。又或许是我太笨没找到...

究竟是什么导致了这个问题,我该如何解决?我发现的所有内容都与访问外部 SQL 服务器有关,但我只是希望 EF 在本地创建一个 .mdf 并连接到它。

我对 Entity Framework 和一般的数据库还很陌生,但我有一些建议可能有助于缩小问题的范围。

  1. "Verify that the instance name is correct and that SQL Server is configured to allow remote connections." 使用不同的程序连接到服务器,看看问题是否出在服务器上。如果您在 EF 外部连接时遇到问题,您可以排除 EF 的问题。

  2. 如果可以,请省略连接字符串或提及您的 SQL 数据库,然后查看 EF 是否会为您生成 LocalDB。 I believe this is the default behaviour of EF. 如果这有效,我认为这也表明您的 SQL 服务器有问题给您带来麻烦。

同样,我对此很陌生,所以我只是想建议我尝试用我有限的知识解决问题的方法。希望对您有所帮助!

好吧,我是偶然想出来的。我在控制台项目中通过 NuGet 安装了 EntityFramework,瞧瞧,Visual Studio 在启动调试会话时创建了 .mdf。

因为它在卸载 EntityFramework 后仍然有效,所以我重新创建了项目并检查了差异。我发现 app.config 中的 EntityFramework 条目仍然存在,所以我将它们复制到新项目中,现在也能正常工作了。

因此,如果您遇到同样的问题,请尝试将这些部分添加到您的控制台项目(或您用来访问数据库 class 库的任何项目类型)的 app.config 中:

<configSections>
  <section name="entityFramework" 
           type="System.Data.Entity.Internal.ConfigFile.EntityFrameworkSection, EntityFramework, Version=6.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089"
           requirePermission="false" />
</configSections>
<connectionStrings>
  <add name="LanguageCreator.data.LanguageCreatorContext"
       connectionString="Data Source=(LocalDB)\v11.0;AttachDbFilename=|DataDirectory|\MyContext.mdf;Integrated Security=SSPI"
       providerName="System.Data.SqlClient" />
</connectionStrings>
<entityFramework>
  <defaultConnectionFactory type="System.Data.Entity.Infrastructure.LocalDbConnectionFactory, EntityFramework">
    <parameters>
      <parameter value="mssqllocaldb" />
    </parameters>
  </defaultConnectionFactory>
  <providers>
    <provider invariantName="System.Data.SqlClient"
              type="System.Data.Entity.SqlServer.SqlProviderServices, EntityFramework.SqlServer" />
  </providers>
</entityFramework>