0

我有这个查询要运行:

string query = @"SELECT *
                 FROM   hint 
                 WHERE  addedSessionId IN (x, y, z, ............)";


if (_connSource.State != ConnectionState.Open)
    _connSource.Open();

MySqlCommand cmd = new MySqlCommand(query, _connSource);
MySqlDataReader r = cmd.ExecuteReader();

List<Hint> lstHint = new List<Hint>();
while (r.Read())
{
    Hint h = new Hint(Convert.ToInt32(r[0]), Convert.ToInt32(r[1]), 
                      Convert.ToString(r[2]), Convert.ToInt32(r[3]), 
                      Convert.ToInt32(r[4]));
    h.addedSessionId = (Guid)r[5];

    lstHint.Add(h);
}

r.Close(); //important

在上面的代码中,如何将值传递x, y, z etc给查询本身?x, y, z etc不是整数或字符串,而是字节数组。在我的代码中,它们是 .net Guid 字段,但我通过将其转换为字节数组将其作为二进制值保存在 db 中。

我可以通过以参数化方式执行此操作来实现我想要的,如下所示:

string query = @"SELECT *
                 FROM   hint 
                 WHERE  addedSessionId = @newSessionId";


if (_connSource.State != ConnectionState.Open)
    _connSource.Open();

List<Hint> lstHint = new List<Hint>();
foreach (List<Guid> myGuid in lstGuid)
{
    MySqlCommand cmd = new MySqlCommand(query, _connSource);
    cmd.Parameters.AddWithValue("newSessionId", myGuid.ToByteArray());
    MySqlDataReader r = cmd.ExecuteReader();

    while (r.Read())
    {
        int id = Convert.ToInt32(r[0]);

        if (IsThisEntryAlreadyAdded(id, lstHint))
            continue;

        Hint h = new Hint(id, Convert.ToInt32(r[1]),
                          Convert.ToString(r[2]), Convert.ToInt32(r[3]),
                          Convert.ToInt32(r[4]));
        h.addedSessionId = (Guid)r[5];

        lstHint.Add(h);
    }

    r.Close(); //important
}

第二种方法的问题是它相对慢得多。这里不仅查询必须多次运行到数据库,而且每次我都需要通过运行此IsThisEntryAlreadyAdded函数来确保该特定条目是否尚未添加。

我的问题是如何在非参数化查询中传递对象(在我的情况下是字节数组)?如果不可能,我的问题是有没有其他方法可以让我的查询更快?

4

2 回答 2

1

这是示例(未仅测试伪代码)我想如何做到这一点:

string query = @"SELECT * FROM   hint  WHERE  addedSessionId IN (";
MySqlCommand cmd = new MySqlCommand(query, _connSource);
int i = 0;
foreach (List<Guid> myGuid in lstGuid)
{
    query = string.Format("{0}@param{1}", query, i);
    cmd.Parameters.AddWithValue(string.Format("@param{0}", i), myGuid.ToByteArray());
    i++;
    if(i != lstGuid.Count) query = string.Format("{0},", query);
}
query = string.Format("{0})", query);
cmd.CommandText = query;
//Here you have command with constructed query and params
于 2012-04-17T12:18:17.043 回答
1

@Reinuz 的回答回答了我的问题,但在我的情况下它很痛苦。当我有非常多的IN (子句值时,我可以超过允许的最大数据包大小,因此查询根本不会运行。

所以这就是我最终得到的结果,这对于大表来说要快得多:

string query = @"SELECT * 
                 FROM   hint 
                 WHERE  addedSessionId IN (" + GetStringValuesFromGuidList()  + ")";

void GetStringValuesFromGuidList()
{
    string guids = null;
    for (int i = 0; i < lstGuid.Count; i++)
    {
        guids += "'" + UTF8Encoding.Default.GetString(lstGuid[i].ToByteArray()).Replace("\\", "\\\\").Replace("'", "''") + "'";
        if (i < lstGuid.Count - 1)
            guids += ", ";
    }

    return guids;
}

我将 Guid 转换为其字符串表示形式,然后将其传递给 MySQL 可以在哪里进行等效字符串搜索的查询。

对于有限数量的条目,参数化查询(Reinuz 的答案)是最好的。对于非常大的值,请进行字符串搜索

Edit: The UTF8Encoding one should use here depends on the character set of the database whether its unicode etc. I had success with .Default in most cases. Furthermore the string representation of binary will have reserved characters. So escape it with ' or \

于 2012-04-20T18:55:52.813 回答