使用Entity Framework 6的SQLite出现"no such table"错误

6
我正在尝试使用SQLiteEntityFramework 6的代码优先方式来实现这个问题。我可以创建数据库并且有一个文件(虽然为空),但是当我尝试向数据库添加条目时,它会说没有这样的表。然而,当我尝试手动创建表格(db.Database.ExecuteSqlCommand("CREATE TABLE etc.))时,它告诉我该表已经存在。
我已经找到了一些关于这个问题的解答,但它们都是关于路径不是绝对路径的问题,但这在这里显然不是问题。

上下文

//Context:
class MediaContext : DbContext
{
    public DbSet<MediaModel> Media { get; set; }

    public MediaContext(string filename): base(new SQLiteConnection() {
        ConnectionString =
            new SQLiteConnectionStringBuilder()
            { DataSource = filename, ForeignKeys = false }
            .ConnectionString
    }, true)
    {

    }        
}

DAL

//In the DAL:
public DataAccessLayer(string path)
{
   _path = Path.GetFullPath(path); //<--The path already comes full in, I'm just paranoid at this point.
   _connectionPath = _path + @"\MyDatabase.sqlite";

   using (var db = new MediaContext(_connectionPath))
   {
      db.Database.CreateIfNotExists();
      db.Database.Initialize(false);                

       db.SaveChanges();
   }

   public void Generate()
   {
       using (var db = new MediaContext(_connectionPath))
       {
          db.Media.Add(new MediaModel("test"));
          db.SaveChanges(); //<--SQLiteException: SQL logic error or missing database: no such table: MediaModels

        }
    }

App.Config

//App.config:
<configuration>
  <configSections>
    <section name="entityFramework" type="System.Data.Entity.Internal.ConfigFile.EntityFrameworkSection, EntityFramework, Version=6.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089" requirePermission="false" />
    <!-- For more information on Entity Framework configuration, visit http://go.microsoft.com/fwlink/?LinkID=237468 -->
  </configSections>
  <startup>
    <supportedRuntime version="v4.0" sku=".NETFramework,Version=v4.5.2" />
  </startup>
  <entityFramework>
    <defaultConnectionFactory type="System.Data.Entity.Infrastructure.LocalDbConnectionFactory, EntityFramework">
      <parameters>
        <parameter value="v13.0" />
      </parameters>
    </defaultConnectionFactory>
    <providers>
      <provider invariantName="System.Data.SQLite.EF6" type="System.Data.SQLite.EF6.SQLiteProviderServices, System.Data.SQLite.EF6" />
      <provider invariantName="System.Data.SqlClient" type="System.Data.Entity.SqlServer.SqlProviderServices, EntityFramework.SqlServer" />
    </providers>
  </entityFramework>
  <system.data>
    <DbProviderFactories>
      <remove invariant="System.Data.SQLite.EF6" />
      <add name="SQLite Data Provider (Entity Framework 6)" invariant="System.Data.SQLite.EF6" description=".NET Framework Data Provider for SQLite (Entity Framework 6)" type="System.Data.SQLite.EF6.SQLiteProviderFactory, System.Data.SQLite.EF6" />
      <remove invariant="System.Data.SQLite.EF6" />
      <add name="SQLite Data Provider" invariant="System.Data.SQLite.EF6" description=".NET Framework Data Provider for SQLite" type="System.Data.SQLite.SQLiteFactory, System.Data.SQLite" />
    </DbProviderFactories>
  </system.data>
</configuration>

这对我的问题没有任何改变。 :/ - Gregor A. Lamche
尝试使用SQLite Browser应用程序查看表是否真实存在,以及它是否具有正确的模式。 - Martin Zikmund
1
如果我用Notepad++打开文件,它完全是空的(0kB)。我曾经设法创建过它,但现在我无法再次复制这个过程(所以我无法检查那个)。然后它有8kB,并且填充了许多NUL字符。因此,我猜测这是创建和保存表格的问题,但正如问题中提到的那样,我也无法手动创建它。 - Gregor A. Lamche
5个回答

25

我已经使Code First方法可用,灵感来自于这篇文章:
使用Entity Framework Core Code First创建SQLite数据库

关键是在上下文构造函数中调用EnsureCreated:

    public MediaContext()
    {
        Database.EnsureCreated();
    }

关于表名与属性名的匹配:
您可以使用 Table 属性来使其生效:

public DbSet<MediaModel> Media { get; set; }

在您的模型类中明确指定表名,即使这似乎是冗余的:

[Table("MediaModel")]
public class MediaModel

EnsureCreated 是核心版本独有的,但 EF6 并没有。 - Dmitry Sokolov
2
如果您正在使用带有新的SqliteConnection("DataSource=:memory:")连接的迁移,则迁移将被忽略,您必须使用"Database.EnsureCreated()"。谢谢+1 - AZ_

8

SQLite的EF6提供程序不支持Code First / Migrations工作流程,因此您必须在EF6工作流程之外手动创建数据库。


我尝试过了,甚至将您的评论标记为答案,但是尽管现在文件中有数据,并且似乎填充了正常的查询,但我仍然无法使用它与EF6。 它与以前一样出现了相同的错误:“没有这样的表:MediaModels”。 令我困扰的是,我甚至没有要求“MediaModels”表的任何地方。 我的上下文可能有问题吗? - Gregor A. Lamche
1
你可能需要在上下文中使用流畅映射API或Table属性明确指定表名。 - ErikEJ

6
这是Microsoft推荐的回复:
Visual Studio在运行.NET Core控制台应用程序时使用了不一致的工作目录。(请参见dotnet/project-system#3619) 这会导致抛出异常:no such table: Blogs. 要更新工作目录: 右键单击项目,选择编辑项目文件 在TargetFramework属性下方添加以下内容: <StartWorkingDirectory>$(MSBuildProjectDirectory)</StartWorkingDirectory> 保存文件 现在您可以运行该应用程序: 调试>开始而不调试

抱歉,但这个不起作用。问题仍然存在。 在https://learn.microsoft.com/en-us/ef/core/get-started/index?tabs=visual-studio中有相同的评论,但它不起作用。 - Jan Wijninckx

3

正如ErikEJ所说的那样,SQLite不支持我使用的代码优先方式,而且显然它有一个严格的命名策略。

为了使所有工作正常,我不得不手动创建表,并将上下文适应命名模式(Media => MediaModels)。

上下文

public DbSet<MediaModel> MediaModels { get; set; }

DAL

public DataAccessLayer(string path)
{
    _path = Path.GetFullPath(path);
    _connectionPath = _path + @"\MyDatabase.sqlite";
    
    using (var db = new MediaContext(_connectionPath))
    {
        db.Database.ExecuteSqlCommand("CREATE TABLE IF NOT EXISTS 'MediaModels' ('Name' TEXT, 'FilePath' TEXT NOT NULL PRIMARY KEY, 'Tags' TEXT, 'Note' TEXT, 'FileExtension' TEXT)");

        db.SaveChanges();
    }
}
    
public void Generate()
{
    using (var db = new MediaContext(_connectionPath))
    {
        db.MediaModels.Add(new MediaModel("test"));
        db.SaveChanges();
    }
}

那段代码还不完美,我会尽力更新这个答案使它更好。


-1

我正在使用.NET Core和EntityFramework Core 2以及SQLite的代码优先迁移。 在OnModelCreating中指定表名有助于解决问题:

    protected override void OnModelCreating(ModelBuilder modelBuilder)
    {
        modelBuilder.Entity<File>()
            .ToTable(typeof(File).Name + "s") // <-- the argument could be just "Files"
            .HasKey(lf => new { lf.MachineName, lf.PathName });

        base.OnModelCreating(modelBuilder);
    }

另一个选择是 - TableAttribute 也应该有所帮助:
    [Table("Files")]
    public class File
    {
        // ...
    }

OnModelCreating 如果同时指定了 TableAttribute 和表名,则会覆盖 TableAttribute 中指定的表名。


网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接