我在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中运行时确实有效。
答案 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;
}
null
值总是明智的,以便优雅地处理它们。cmd.CommandType
将告诉ADO.NET它将如何执行您的命令。using
块,以便在超出范围时处理您的资源。using Statement (C# Reference)
提供方便的语法,确保正确使用
IDisposable
个对象。
提供释放非托管资源的机制。
由于您的程序似乎运行正常,除非您另有说明,否则我无法弄清楚其他错误。
dr.HasRows == false
第一次看到您的SQL查询时,我的查询超时并不奇怪。
来自MSDN:
IDbCommand.CommandTimeout Property
获取或设置终止执行命令并生成错误的等待时间。
Remarks (SqlCommand.CommandTimeout Property)
值为0表示没有限制(尝试执行命令将无限期地等待)。
我看到你试图设置CommandTimeout = 0
。你永远不会那样做,除了诊断你的查询是否需要太长时间。
相反,您应考虑将超时属性设置为您的软件性能要求可接受的属性。尝试确定执行所需的时间,并相应地设置 CommandTimeout 。
默认值为30秒。也许您的查询需要45秒或更长时间才能执行。找到有效的方法,这将允许您设置正常执行时间。然后,当需要更长时间时,这意味着有些错误不一定是您的代码。
此外,尝试确定您的查询是否可以使用更多索引等,通过SSAS分析其执行计划,这将指出您的查询需要最多时间。