2

下面的 C# 代码检查 SQL 数据库以查看记录是否与 ClientID 和用户名匹配。如果找到超过 15 条或更多匹配的记录,则我的 Windows 2008 服务器上的 CPU 峰值约为 78%,而在执行以下 C# 代码时会找到 15 条记录。SQL Server 2008 数据库和软件位于另一台服务器上,因此问题不在于 SQL Server 使 CPU 达到峰值。问题在于我的 C# 软件正在执行下面的代码。在执行数据库查询并找到记录时,我可以看到包含以下 C# 代码的软件可执行文件飙升至 78%。

有人可以告诉我,当找到 15 个或更多匹配记录时,我的代码是否有问题导致 CPU 出现峰值?你能告诉我/告诉我如何优化我的代码吗?

更新:如果它找到 10 条记录,CPU 只会以 2-3% 的速度出现峰值。只有当它找到 15 条或更多记录时,CPU 才会以 78% 的速度飙升 2 到 3 秒。

//ClientID[0] will contain a ClientID of 10 characters
//output[0] will contain a User Name
char[] trimChars = { ' ' };
using (var connection = new SqlConnection(string.Format(GlobalClass.SQLConnectionString, "History")))
{
    connection.Open();
    using (var command = new SqlCommand())
    {
        command.CommandText = string.Format(@"SELECT Count(*) FROM Filelist WHERE [ToAccountName] = '" + output[0] + @"'");
        command.Connection = connection;
        var rows = (int) command.ExecuteScalar();
        if (rows >= 0)
        {
            command.CommandText = string.Format(@"SELECT * FROM Filelist WHERE [ToAccountName] = '" + output[0] + @"'");
            using (SqlDataReader reader = command.ExecuteReader())
            {
                if (reader.HasRows)
                {
                    while (reader.Read())
                    {
                        //Make sure ClientID does NOT exist in the ClientID field
                        if (reader["ClientID"].ToString().TrimEnd(trimChars).IndexOf(ClientID[0]) !=
                            -1)
                        {
                            //If we are here, then do something
                        }
                    }
                }
                reader.Close();
                reader.Dispose();
            }
        }
        // Close the connection
        if (connection != null)
        {
            connection.Close();
        }
    }
}
4

7 回答 7

4

如果要删除第一个查询,您可以将数据库访问次数从 2 减少到 1,这不是必需的。

using (SqlConnection connection = new SqlConnection(connectionString))
using (SqlCommand command = connection.CreateCommand())
{
    command.CommandText = "SELECT ClientID FROM dbo.Filelist WHERE ToAccountName = @param"; // note single column in select clause
    command.Parameters.AddWithValue("@param", output[0]); // note parameterized query

    connection.Open();
    using (SqlDataReader reader = command.ExecuteReader())
    {  
        while (reader.Read()) // reader.HasRow is doubtfully necessary
        {
            // logic goes here
            // but it's better to perform it on data layer too

            // or return all clients first, then perform client-side logic
            yield return reader.GetString(0);
        }
    } // note that using block calls Dispose()/Close() automatically
}
于 2012-11-17T17:13:59.297 回答
3

改变这个:

SELECT * FROM Filelist

对此:

SELECT ClientID FROM Filelist

并检查性能。我怀疑您的选择中有一个 blob 字段。也不建议在select *查询中写下您确切感兴趣的字段。

于 2012-11-17T17:11:17.940 回答
3

没有什么看起来明显占用 CPU 资源,但一个问题确实很突出。

您正在运行查询以计算有多少条记录

"SELECT Count(*) FROM Filelist WHERE [ToAccountName] = '" + output[0] + @"'"

然后,如果返回大于 0,则您正在运行另一个查询来获取数据。

"SELECT * FROM Filelist WHERE [ToAccountName] = '" + output[0] + @"'"

这是多余的。摆脱第一个查询,只使用第二个,检查阅读器是否有数据。你也可以摆脱 HasRows 调用,然后做

using (SqlDataReader reader = command.ExecuteReader())
{
    while (reader.Read())
    {
    }
}
于 2012-11-17T17:23:42.507 回答
1

请考虑关于参数化查询的内容。

除此之外,我认为唯一的大问题可能出现在以下块中:

while (reader.Read())
{
    //Make sure ClientID does NOT exist in the ClientID field
    if (reader["ClientID"].ToString().TrimEnd(trimChars).IndexOf(ClientID[0]) != -1)
    {
        //If we are here, then do something
    }
}

因此,尝试将您的 reader.Read() 数据缓存在某个局部变量中,尽快释放 SQL 资源,然后您就可以处理刚刚检索到的数据。例如:

List<string> myRows = new List<string>();
while (reader.Read())
{
   myRows.Add(reader["ClientID"].ToString();
}
/// quit the using clause
/// now elaborate what you got in myRows
于 2012-11-17T17:12:51.593 回答
0

代码中没有任何内容表明存在性能问题。

SQL Profiler 显示什么?

(在查询计划和使用的服务器资源方面。)

编辑:为了使这一点更清楚:您有一个可能表明存在问题的测量值。您现在需要更深入地测量以了解它是否真的是一个问题,只有您可以这样做(没有其他人可以访问硬件)。

于 2012-11-17T17:05:10.653 回答
0

我建议按照建议使用参数,但是,我看到了字符串列的类型与 C# 字符串不匹配的性能问题。在这些情况下,我建议明确指定类型。

像这样:

command.CommandText = "SELECT ClientID FROM dbo.Filelist WHERE ToAccountName = @accountName"; 
command.Parameters.Add("@accountName", SqlDbType.NVarChar, 16, output[0]);

或这个:

SqlParameter param = command.Parameters.Add(
    "@accountName", SqlDbType.NVarChar);
param.Size = 16; //optional
param.Value = output[0];
于 2012-11-17T18:03:25.273 回答
0

我强烈建议您从 JetBrains获取一份dotTrace 。

至少,分析客户端代码将帮助您识别/消除 CPU 峰值的来源。

于 2012-11-17T17:25:53.023 回答