错误:使用ExecuteNonQuery()时“无法在datareader处于活动状态时设置commandtext”

时间:2011-09-29 09:10:48

标签: c# .net sqlite system.data.sqlite

我监听数据流并将数据作为插入语句存储在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无关。

如何摆脱这个错误?

1 个答案:

答案 0 :(得分:12)

我会为每个SQL语句创建一个新的SqlCommand(或者sqlCommand的任何类型)。让连接池处理使其全部高效 - 每次您需要对数据库执行某些操作时:

  • 创建并打开连接
  • 创建命令
  • 执行命令
  • 处置命令和连接(使用using语句)

这样,除了由于连接池空间不足等原因之外,你不能最终得到影响另一个命令的一个命令的本地状态。

开发人员经常尝试通过一次使用一个连接(以及可能的命令)进行优化,但这是一种虚假的经济,并导致像你所展示的那样的问题。