C#程序没有从SQL查询中获得结果

时间:2014-10-21 01:48:39

标签: c# sql-server tsql

我在C#中编写了一个函数来检索TSQL查询的结果。

public List<string> ExecuteRule(string rule)
{
    List<string> foundLCNs = new List<string>();
    try
    {
        myConnection = new SqlConnection(ConnectionString);

        SqlDataReader myReader = null;
        SqlCommand myCommand = new SqlCommand
        (
            rule, 
            myConnection 
        ); 
        myConnection.Open();

        myReader = myCommand.ExecuteReader();
        while(myReader.Read())
        {
            foundLCNs.Add(myReader["labno"].ToString());
        }
    }
    catch (Exception e)
    {
        Controller.PrintStackTrace (e);
        Console.WriteLine(e.ToString());
    }
    finally
    {
        myConnection.Close();
    }
    return foundLCNs;
}

我发现当我运行一些SQL查询时,没有返回任何结果,但是当我复制SQL并在Microsoft SQL Server管理sudio中运行它时,我得到了一些结果。

也许查询可能花了太长时间,但我没有得到任何例外。

当我使用调试器逐步执行代码时 我点击了这一行

  

myReader = myCommand.ExecuteReader();

快速执行(检索数据需要几秒钟)。

while循环从不执行内部代码,它只是跳过,因为没有返回数据。

修改

SQL看起来像这样

WITH FilterPatches AS
(
    SELECT 
        OIL_SAMPLE.labno 
    FROM OIL_SAMPLE 
    INNER JOIN LU_OIL_TEST_TYPE ON OIL_SAMPLE.test_type_auto = LU_OIL_TEST_TYPE.test_type_auto 
    LEFT JOIN OIL_EQ_UNIT ON OIL_SAMPLE.equnit_auto = OIL_EQ_UNIT.equnit_auto 
    LEFT JOIN Equipment ON OIL_EQ_UNIT.equipmentid_auto = Equipment.equipmentid_auto 
    LEFT JOIN CRSF ON Equipment.crsf_auto = CRSF.crsf_auto 
    LEFT JOIN CUSTOMER ON CRSF.customer_auto = CUSTOMER.customer_auto 
    LEFT JOIN OIL_SAMPLE_READING_STORE AS DEP ON OIL_SAMPLE.labno = DEP.labno 
    LEFT JOIN OIL_SAMPLE_READING AS DEP2 ON OIL_SAMPLE.labno_auto = DEP2.labno_auto 
    WHERE CUSTOMER.custid IN 
    ( 
        '555555', '555'
    ) AND 
    ( 
        (
            DEP.element_auto IN 
            ( 
                 SELECT element_auto 
                 FROM LU_OIL_ELEMENT 
                 WHERE LU_OIL_ELEMENT.elementid = 'DEP' 
            ) 
            AND 
            DEP.reading NOT IN ('vw', 'ok') 
        ) 
        OR 
        ( 
            DEP2.element_auto IN 
            ( 
                SELECT element_auto 
                FROM LU_OIL_ELEMENT 
                WHERE LU_OIL_ELEMENT.elementid = 'DEP' 
            ) 
            AND 
            DEP2.reading NOT IN ('vw', 'ok') 
            AND OIL_SAMPLE.data_released_date IS NULL
        )
    ) 
    AND OIL_SAMPLE.labno NOT IN
    ( 
        SELECT labno 
        FROM OIL_SAMPLE_READING_STORE 
        WHERE element_auto IN 
        ( 
            SELECT element_auto 
            FROM LU_OIL_ELEMENT 
            WHERE LU_OIL_ELEMENT.elementid = 'FPATCH' 
        ) 
    ) 
    AND LU_OIL_TEST_TYPE.test_type_desc NOT IN 
    (
        'Diesel',
        'DieselStd',
        'DieselSml'
    )
),
ParticleCountElements AS
(
    SELECT element_auto 
    FROM LU_OIL_ELEMENT 
    WHERE LU_OIL_ELEMENT.elementid IN
    (
        'AB',
        'CD'
    ) 
)
(

    SELECT labno
    FROM FilterPatches

    EXCEPT

    (
        SELECT DISTINCT labno
        FROM OIL_SAMPLE_READING_STORE
        WHERE element_auto IN
        (
            SELECT element_auto
            FROM ParticleCountElements
        )
        AND labno IN 
        (
            SELECT labno
            FROM FilterPatches
        )
        AND reading_raw IS NOT NULL

        UNION

        SELECT labno
        FROM OIL_SAMPLE 
        LEFT JOIN OIL_SAMPLE_READING ON OIL_SAMPLE.labno_auto = OIL_SAMPLE_READING.labno_auto 
        WHERE OIL_SAMPLE.data_released_date IS NULL
        AND OIL_SAMPLE_READING.element_auto IN
        (
            SELECT element_auto
            FROM ParticleCountElements
        )
        AND labno IN 
        (
            SELECT labno
            FROM FilterPatches
        )

    )
)

我已经尝试在microsoft sql server management studio中运行它,它运行正常。

修改

按照意见建议我尝试将代码更改为

List<string> foundLCNs = new List<string>();

using (var cnx = new SqlConnection (ConnectionString)) 
{
    using (SqlCommand cmd = new SqlCommand (rule, cnx)) 
    {
        cmd.CommandType = CommandType.Text;
        cnx.Open ();

        using (var dr = cmd.ExecuteReader())
        {
            if (dr.HasRows) 
            {
                while (dr.Read ()) 
                {
                    if (!dr.IsDBNull (0)) 
                    {
                        foundLCNs.Add (dr.GetString (0));
                    }
                }
            }
        }
    }
}

return foundLCNs;
  

dr.HasRows   是假的

我仍然有同样的行为。它适用于其他查询,但不适用于我发布的查询。但是,此查询在SQL Server中运行时确实有效。

1 个答案:

答案 0 :(得分:2)

以下是您的代码看起来相似的方式。

public IList<string> ExecuteRule(string rule) {
    var found = new List<string>();

    using (var cnx = new SqlConnection(connectionString)) 
        using (var cmd = new SqlCommand(rule, cnx)) {
            cmd.CommandType = CommandType.Text;

            // Make sure to add values to parameters whenever required by your query.
            // e.g. cmd.Parameters.AddWithValue("@paramName", value);

            cnx.Open();

            using (var dr = cmd.ExecuteReader()) 
                if (dr.HasRows)
                    while (dr.Read()) 
                        if (!dr.IsDBNull(0)) 
                            found.Add(dr.GetStringValue(0));                                        
        }   

    return found;         
}
  1. 检查null值总是明智的,以便优雅地处理它们。
  2. 设置cmd.CommandType将告诉ADO.NET它将如何执行您的命令。
  3. 提供适当的参数(如果有的话)是强制性的(但事实并非如此,因为它运行时没有错误)。
  4. 始终使用using块,以便在超出范围时处理您的资源。
  5. using Statement (C# Reference)

      

    提供方便的语法,确保正确使用IDisposable个对象。

    IDisposable

      

    提供释放非托管资源的机制。

    由于您的程序似乎运行正常,除非您另有说明,否则我无法弄清楚其他错误。


    dr.HasRows == false

    第一次看到您的SQL查询时,我的查询超时并不奇怪。

    来自MSDN:

    IDbCommand.CommandTimeout Property

      

    获取或设置终止执行命令并生成错误的等待时间。

         

    Remarks (SqlCommand.CommandTimeout Property)

         

    值为0表示没有限制(尝试执行命令将无限期地等待)。

    我看到你试图设置CommandTimeout = 0。你永远不会那样做,除了诊断你的查询是否需要太长时间。

    相反,您应考虑将超时属性设置为您的软件性能要求可接受的属性。尝试确定执行所需的时间,并相应地设置 CommandTimeout

    默认值为30秒。也许您的查询需要45秒或更长时间才能执行。找到有效的方法,这将允许您设置正常执行时间。然后,当需要更长时间时,这意味着有些错误不一定是您的代码。

    此外,尝试确定您的查询是否可以使用更多索引等,通过SSAS分析其执行计划,这将指出您的查询需要最多时间。