我已经开发了许多.NET / SQL Server应用程序,但是我遭受了SQL查询超时的困扰,无法了解到底。我在查找有问题的查询并重新索引/重新编写它们方面有丰富的经验。我的Web应用程序使用SQL Server的RDS和Web应用程序的EC2托管在AWS上。我们每天有100-200个唯一身份用户,数据库约为15GB,其中有两个表大于1GB。
我全天都会看到以下异常消息:
'Execution Timeout Expired. The timeout period elapsed prior to completion of the operation or the server is not responding.'
遭受超时的查询与发生超时的时间一样随机。它似乎与任何显而易见的事情(备份运行一整夜等)不符。
我尝试从C#应用程序获取每个查询,然后直接在SQL中运行(使用与Arith Abort相同的SET选项),它们都运行良好。有些查询本质上是较慢的查询,但是最慢的查询在大约2秒钟内运行,并且具有约400k逻辑读取。但是,我还看到查询超时在15毫秒内运行,并且具有<10个逻辑读取。
我看到的最奇怪的事情是我已经从Web应用程序中获取了一个查询,并将其编码到已运行24小时的控制台应用程序中,每秒调用一次查询。即使我已经看到主系统在运行期间同一查询的超时,它也没有一个异常/超时。
我最近将RDS服务器升级到了M5 Large,并且每天都在一夜之间重建所有索引。我已经运行DBCC FREEPROCCACHE,以确保没有造成问题的过时查询计划。
我觉得这是参数嗅探,或者我最后的想法是硬件/网络故障,但这确实吸引了稻草人!
我得到的堆栈跟踪看起来像是中间查询,而不是在连接阶段。
at System.Data.SqlClient.SqlInternalConnection.OnError(SqlException exception, Boolean breakConnection, Action`1 wrapCloseInAction)
at System.Data.SqlClient.TdsParser.ThrowExceptionAndWarning(TdsParserStateObject stateObj, Boolean callerHasConnectionLock, Boolean asyncClose)
at System.Data.SqlClient.TdsParserStateObject.ReadSniError(TdsParserStateObject stateObj, UInt32 error)
at System.Data.SqlClient.TdsParserStateObject.ReadSniSyncOverAsync()
at System.Data.SqlClient.TdsParserStateObject.TryReadNetworkPacket()
at System.Data.SqlClient.TdsParserStateObject.TryPrepareBuffer()
at System.Data.SqlClient.TdsParserStateObject.TryReadByteArray(Byte[] buff, Int32 offset, Int32 len, Int32& totalRead)
at System.Data.SqlClient.TdsParserStateObject.TryReadString(Int32 length, String& value)
at System.Data.SqlClient.TdsParser.TryReadSqlStringValue(SqlBuffer value, Byte type, Int32 length, Encoding encoding, Boolean isPlp, TdsParserStateObject stateObj)
at System.Data.SqlClient.TdsParser.TryReadSqlValue(SqlBuffer value, SqlMetaDataPriv md, Int32 length, TdsParserStateObject stateObj, SqlCommandColumnEncryptionSetting columnEncryptionOverride, String columnName)
at System.Data.SqlClient.SqlDataReader.TryReadColumnInternal(Int32 i, Boolean readHeaderOnly)
at System.Data.SqlClient.SqlDataReader.TryReadColumn(Int32 i, Boolean setTimeout, Boolean allowPartiallyReadColumn)
at System.Data.SqlClient.SqlDataReader.GetValueInternal(Int32 i)
at System.Data.SqlClient.SqlDataReader.GetValue(Int32 i)
在某种程度上解决问题的任何技术帮助将不胜感激,我担心它会突然变得更糟。
谢谢
编辑1
我试图通过每10ms运行一次测试应用程序(如上)并同时在SSMS中运行缓慢的阻塞事务来在本地创建相同的问题。
从应用查询
SELECT TOP 10 *
FROM MyTable
WHERE LastModifiedBy = 'Stu'
在SSMS中查询
BEGIN TRAN
UPDATE TOP (10000) MyTable SET LastModifiedBy = 'Me' where LastModifiedBy = 'Me'
WAITFOR DELAY '00:00:35'
COMMIT
当出现此错误时,我会看到我通常希望在SQL Profiler中看到的内容,其中应用程序查询恰好花费30000ms,并且在应用程序中出现异常。但是,此操作的有用输出是堆栈跟踪与我在生产中(上图)看到的不同。
at System.Data.SqlClient.SqlConnection.OnError(SqlException exception, Boolean breakConnection, Action`1 wrapCloseInAction)
at System.Data.SqlClient.SqlInternalConnection.OnError(SqlException exception, Boolean breakConnection, Action`1 wrapCloseInAction)
at System.Data.SqlClient.TdsParser.ThrowExceptionAndWarning(TdsParserStateObject stateObj, Boolean callerHasConnectionLock, Boolean asyncClose)
at System.Data.SqlClient.TdsParser.TryRun(RunBehavior runBehavior, SqlCommand cmdHandler, SqlDataReader dataStream, BulkCopySimpleResultSet bulkCopyHandler, TdsParserStateObject stateObj, Boolean& dataReady)
at System.Data.SqlClient.SqlDataReader.TryConsumeMetaData()
at System.Data.SqlClient.SqlDataReader.get_MetaData()
at System.Data.SqlClient.SqlCommand.FinishExecuteReader(SqlDataReader ds, RunBehavior runBehavior, String resetOptionsString, Boolean isInternal, Boolean forDescribeParameterEncryption, Boolean shouldCacheForAlwaysEncrypted)
at System.Data.SqlClient.SqlCommand.RunExecuteReaderTds(CommandBehavior cmdBehavior, RunBehavior runBehavior, Boolean returnStream, Boolean async, Int32 timeout, Task& task, Boolean asyncWrite, Boolean inRetry, SqlDataReader ds, Boolean describeParameterEncryptionRequest)
at System.Data.SqlClient.SqlCommand.RunExecuteReader(CommandBehavior cmdBehavior, RunBehavior runBehavior, Boolean returnStream, String method, TaskCompletionSource`1 completion, Int32 timeout, Task& task, Boolean& usedCache, Boolean asyncWrite, Boolean inRetry)
at System.Data.SqlClient.SqlCommand.RunExecuteReader(CommandBehavior cmdBehavior, RunBehavior runBehavior, Boolean returnStream, String method)
at System.Data.SqlClient.SqlCommand.ExecuteReader(CommandBehavior behavior, String method)
at System.Data.SqlClient.SqlCommand.ExecuteDbDataReader(CommandBehavior behavior)
at System.Data.Common.DbCommand.System.Data.IDbCommand.ExecuteReader(CommandBehavior behavior)
我正在读取此堆栈跟踪,因为查询从未开始执行,因为它仍在尝试读取查询的元数据。但是,这与生产中的堆栈跟踪形成了对比(在我看来),该跟踪似乎在从列读取数据的中间,但是执行中途超时。
我也一直在阅读.NET 4.6.2,这是我们正在使用的版本。我将在今天晚上将所有内容升级到4.7.2,以排除这种情况。 (Connection to remote SQL server breaks when upgrading web server to .net framework 4.6.1)
答案 0 :(得分:0)
经过一周的紧张调查后,问题已解决!!它已经运行了两个多小时,没有一个超时时间:-)
事实证明是与.NET v4.6.2。的某种错误或不匹配。
我的配置为:
我的更改是:
<httpRuntime targetFramework="4.7.2" />
)这些问题帮助我指出了这个方向:
谢谢您的互联网-在您开始学习之前,我是如何编码的