我监听数据流并将数据作为插入语句存储在ConcurrentQueue
中,并使用System.Threading.Timer
插入带有批量插入的数据,间隔为1000。整个场景在静态类上运行。
这是代码:
static void timer_Elapsed(object sender, ElapsedEventArgs e)
{
if (queryQueue.IsEmpty)
return;
string text = "";
//bulkBuilder is StringBuilder.
//queryQueue is ConcurrentQueue
bulkBuilder.AppendLine("PRAGMA synchronous = 0;PRAGMA count_changes = FALSE;PRAGMA journal_mode=OFF;Begin;");
while (queryQueue.TryDequeue(out text))
{
bulkBuilder.Append(text);
bulkBuilder.AppendLine(";");
}
bulkBuilder.AppendLine("Commit;");
try
{
sqlCommand.CommandText = bulkBuilder.ToString();
sqlCommand.ExecuteNonQuery();
}
catch (System.Exception ex)
{
Console.WriteLine("Error while inserting Data : " + ex.Message);
}
finally
{
bulkBuilder.Clear();
}
}
有趣的是,sqlCommand
仅用于插入,仅用于此计时器中的ExecuteNonQuery()
。并且不时会收到错误消息“无法在datareader处于活动状态时设置commandtext”。这是无稽之谈,因为此代码与SQLiteDataReader
中的内部sqlCommand
无关。
如何摆脱这个错误?
答案 0 :(得分:12)
我会为每个SQL语句创建一个新的SqlCommand
(或者sqlCommand
的任何类型)。让连接池处理使其全部高效 - 每次您需要对数据库执行某些操作时:
using
语句)这样,除了由于连接池空间不足等原因之外,你不能最终得到影响另一个命令的一个命令的本地状态。
开发人员经常尝试通过一次使用一个连接(以及可能的命令)进行优化,但这是一种虚假的经济,并导致像你所展示的那样的问题。