2011-04-27 133 views
4

我写了一个简单的应用程序(称为app1),它读取SQLite数据库并在GridView中显示内容。我有一个单独的C#控制台应用程序(app2)需要写入同一个数据库。问题是app2失败,出现“数据库已锁定”错误。我可以看到,只要我启动app1,就会创建一个userdb-journal文件。我认为问题是,app1打开数据库,但不释放它?这是我用于填充绑定到app1中的网格的表的代码。C#:SQLite数据库始终锁定

public DataTable GetAllPeople() 
    { 
     var connectionString = "Data Source=" + dbPath + ";Version=3"; 

     using (SQLiteDataAdapter sqlDataAdapter = 
      new SQLiteDataAdapter("SELECT id,FirstName,LastName,Address FROM Users", 
            connectionString)) 
     { 
      using (DataTable dataTable = new DataTable()) 
      { 
       sqlDataAdapter.Fill(dataTable); 
       // code to add some new columns here 

       return dataTable; 
      } 
     } 
    } 

这里是填充GridView控件的代码:

private void Form1_Load(object sender, EventArgs e) 
    { 
     UserDatabase db = new UserDatabase(); 
     db.Initialize(); 
     dataGridView1.DataSource = db.GetAllPeople(); 

    } 

我怎样才能解决事情,所以APP 2可以读取并同时APP1运行写入数据库?

编辑 看起来像那个日志文件只是由app2创建的。我只注意到app1在运行时数据库锁定错误,但也许app1是一个红色的鲱鱼。 App2是多线程的。也许我应该开始一个关注app2和多线程访问的新问题?

编辑 感谢您的所有意见。我已经锁定了所有数据库访问,并将所有内容都包含在使用中。现在似乎都在工作。

+0

SQLite日志是在您将*写入数据库时​​创建的,不会读取它。而'GetAllPeople'只能从数据库中读取,所以它可能暂时只获得一个单独的共享锁,然后返回。你确定你没有写你的分贝在你的app1? – Groo 2011-04-27 11:29:16

+0

你是对的。我错了。看起来这个日志文件只是由app2创建的。我只注意到app1在运行时数据库锁定错误,但也许app1是一个红色的鲱鱼。 App2是多线程的。也许我应该开始一个关注app2和多线程访问的新问题? – RogerS 2011-04-27 12:14:25

+0

@RogerS:你应该专注于你写入数据库的部分,那就是它被锁定的部分。检查配置的读取超时,并确保您正确处理连接。很可能有一部分代码执行冗长的插入或更新操作,并将文件锁定时间过长。 – Groo 2011-04-27 13:34:18

回答

0

您是否要求SQLITE等待并再次尝试db是否被锁定?以下是如何做到在C

// set SQLite to wait and retry for up to 100ms if database locked 
    sqlite3_busy_timeout(db, 100); 

的一点是,SQLITE锁定DB简单,当它被访问。如果另一个线程或进程在阻塞时访问它,默认情况下SQLITE会返回一个错误。但是,您可以让它等待,并通过上述呼叫自动重试。这解决了许多这类问题。

+0

谢谢,看起来非常有用。你知道如何从C#做到这一点? – RogerS 2011-04-27 13:12:37

+0

@RogerS我不使用C#多。我的猜测是:将'busy_timeout = 100'添加到连接字符串中。 – ravenspoint 2011-04-27 13:49:14

+0

位迟了,但以防万一有人需要它:如果您正在使用System.Data.SQLite.dll,您可以使用SQLiteConnectionStringBuilder类及其DefaultTimeout属性来定义超时。 – 2012-02-24 16:08:30

2

以下是代码,在连接字符串构建器上轻松设置参数,并使用它构建SQLiteConnection。

SQLiteConnectionStringBuilder connBuilder = new SQLiteConnectionStringBuilder(); 
     connBuilder.DataSource = filePath; 
     connBuilder.Version = 3; 
     connBuilder.CacheSize = 4000;    
     connBuilder.DefaultTimeout = 100; 
     connBuilder.Password = "mypass"; 


     using(SQLiteConnection conn = new SQLiteConnection(connBuilder.ToString())) 
     { 
      //... 
     } 

问候。