title | description | author | ms.author | ms.date | ms.service | ms.subservice | ms.topic |
---|---|---|---|---|---|---|---|
Asynchronous programming |
Learn about asynchronous programming in the Microsoft SqlClient Data Provider for SQL Server. |
David-Engel |
v-davidengel |
12/04/2020 |
sql |
connectivity |
conceptual |
[!INCLUDEappliesto-netfx-netcore-netst-md]
[!INCLUDEDriver_ADONET_Download]
This article discusses support for asynchronous programming in the Microsoft SqlClient Data Provider for SQL Server (SqlClient).
The Microsoft SqlClient Data Provider for SQL Server includes methods from System.Data.SqlClient to maintain backwards compatibility for applications migrating to xref:Microsoft.Data.SqlClient. It isn't recommended to use the following legacy asynchronous programming methods for new development:
-
xref:Microsoft.Data.SqlClient.SqlCommand.BeginExecuteNonQuery%2A?displayProperty=nameWithType
-
xref:Microsoft.Data.SqlClient.SqlCommand.BeginExecuteReader%2A?displayProperty=nameWithType
-
xref:Microsoft.Data.SqlClient.SqlCommand.BeginExecuteXmlReader%2A?displayProperty=nameWithType
Tip
In the Microsoft SqlClient Data Provider for SQL Server, these legacy methods no longer require Asynchronous Processing=true
in the connection string.
These asynchronous programming features provide a simple technique to make code asynchronous.
For more information about asynchronous programming in .NET, see:
When your user interface is unresponsive or your server doesn't scale, it's likely you need your code to be more asynchronous. Writing asynchronous code has traditionally involved installing a callback (also called continuation) to express the logic that occurs after the asynchronous operation finishes. This style complicates the structure of asynchronous code as compared with synchronous code.
You can call into asynchronous methods without using callbacks, and without splitting your code across multiple methods or lambda expressions.
The async
modifier specifies that a method is asynchronous. When calling an async
method, a task is returned. When the await
operator is applied to a task, the current method exits immediately. When the task finishes, execution resumes in the same method.
Tip
In the Microsoft SqlClient Data Provider for SQL Server, asynchronous calls are not required to set the Context Connection
connection string keyword.
Calling an async
method doesn't create extra threads. It may use the existing I/O completion thread briefly at the end.
The following methods in the Microsoft SqlClient Data Provider for SQL Server support asynchronous programming:
-
xref:System.Data.Common.DbConnection.OpenAsync%2A?displayProperty=nameWithType
-
xref:System.Data.Common.DbCommand.ExecuteDbDataReaderAsync%2A?displayProperty=nameWithType
-
xref:System.Data.Common.DbCommand.ExecuteNonQueryAsync%2A?displayProperty=nameWithType
-
xref:System.Data.Common.DbCommand.ExecuteReaderAsync%2A?displayProperty=nameWithType
-
xref:System.Data.Common.DbCommand.ExecuteScalarAsync%2A?displayProperty=nameWithType
-
xref:System.Data.Common.DbDataReader.GetFieldValueAsync%2A
-
xref:System.Data.Common.DbDataReader.IsDBNullAsync%2A
-
xref:System.Data.Common.DbDataReader.NextResultAsync%2A?displayProperty=nameWithType
-
xref:System.Data.Common.DbDataReader.ReadAsync%2A?displayProperty=nameWithType
-
xref:Microsoft.Data.SqlClient.SqlConnection.OpenAsync%2A?displayProperty=nameWithType
-
xref:Microsoft.Data.SqlClient.SqlCommand.ExecuteNonQueryAsync%2A?displayProperty=nameWithType
-
xref:Microsoft.Data.SqlClient.SqlCommand.ExecuteReaderAsync%2A?displayProperty=nameWithType
-
xref:Microsoft.Data.SqlClient.SqlCommand.ExecuteScalarAsync%2A?displayProperty=nameWithType
-
xref:Microsoft.Data.SqlClient.SqlCommand.ExecuteXmlReaderAsync%2A?displayProperty=nameWithType
-
xref:Microsoft.Data.SqlClient.SqlDataReader.NextResultAsync%2A?displayProperty=nameWithType
-
xref:Microsoft.Data.SqlClient.SqlDataReader.ReadAsync%2A?displayProperty=nameWithType
-
xref:Microsoft.Data.SqlClient.SqlBulkCopy.WriteToServerAsync%2A?displayProperty=nameWithType
Other asynchronous members support SqlClient streaming support.
Tip
The asynchronous methods don't require Asynchronous Processing=true
in the connection string. And this property is obsolete in the Microsoft SqlClient Data Provider for SQL Server.
You can upgrade an existing application to use the asynchronous feature. For example, assume an application has a synchronous connection algorithm and blocks the UI thread every time it connects to the database. Once connected, the application calls a stored procedure that signals other users of the one who just signed in.
[!code-csharpSqlCommand_ExecuteNonQuery#1]
When converted to use the asynchronous functionality, the program would look like:
[!code-csharpSqlCommand_ExecuteNonQueryAsync#1]
It's also possible to add asynchronous capability (SqlConnection::OpenAsync) without changing the existing asynchronous logic. For example, if an application currently uses:
[!code-csharpSqlConnection_OpenAsync_ContinueWith#1]
You can begin to use the asynchronous pattern without substantially changing the existing algorithm.
[!code-csharpSqlConnection_OpenAsync_ContinueWith#2]
You may need to create a tool that can connect to different databases and execute queries. You can use the base provider model and the asynchronous feature.
The Microsoft Distributed Transaction Controller (MSDTC) must be enabled on the server to use distributed transactions. For information on how to enable MSDTC, see How to Enable MSDTC on a Web Server.
[!code-csharpSqlClient_AsyncProgramming_ProviderModel#1]
[!code-csharpSqlClient_AsyncProgramming_SqlTransaction#1]
In an enterprise application, you may need to add distributed transactions in some scenarios, to enable transactions between multiple database servers. You can use the System.Transactions namespace and enlist a distributed transaction, as follows:
[!code-csharpSqlClient_AsyncProgramming_DistributedTransaction#1]
You can cancel an asynchronous request by using the xref:System.Threading.CancellationToken.
[!code-csharpSqlClient_AsyncProgramming_Cancellation#1]
Asynchronous capabilities are also in xref:Microsoft.Data.SqlClient.SqlBulkCopy?displayProperty=nameWithType with xref:Microsoft.Data.SqlClient.SqlBulkCopy.WriteToServerAsync%2A?displayProperty=nameWithType.
[!code-csharpSqlClient_AsyncProgramming_SqlBulkCopy#1]
The example opens a single connection to the AdventureWorks database. Using a xref:Microsoft.Data.SqlClient.SqlCommand object, a xref:Microsoft.Data.SqlClient.SqlDataReader is created. As the reader is used, a second xref:Microsoft.Data.SqlClient.SqlDataReader is opened, using data from the first xref:Microsoft.Data.SqlClient.SqlDataReader as input to the WHERE clause for the second reader.
Note
The following example uses the sample AdventureWorks database. The connection string provided in the sample code assumes that the database is installed and available on the local computer. Modify the connection string as necessary for your environment.
[!code-csharpSqlClient_AsyncProgramming_MultipleCommands#1]
MARS allows a connection to be used for both read operations and data manipulation language (DML) operations with more than one pending operation. This feature eliminates the need for an application to deal with connection-busy errors. Also, MARS can replace the use of server-side cursors, which generally consume more resources. Finally, because multiple operations can operate on a single connection, they can share the same transaction context, eliminating the need to use sp_getbindtoken and sp_bindsession system stored procedures.
The following Console application demonstrates how to use two xref:Microsoft.Data.SqlClient.SqlDataReader objects with three xref:Microsoft.Data.SqlClient.SqlCommand objects and a single xref:Microsoft.Data.SqlClient.SqlConnection object with MARS enabled. The first command object retrieves a list of vendors whose credit rating is 5. The second command object uses the vendor ID provided from a xref:Microsoft.Data.SqlClient.SqlDataReader to load the second xref:Microsoft.Data.SqlClient.SqlDataReader with all of the products for the particular vendor. Each product record is visited by the second xref:Microsoft.Data.SqlClient.SqlDataReader. A calculation is performed to determine what the new OnOrderQty should be. The third command object is then used to update the ProductVendor table with the new value. This entire process takes place within a single transaction, which is rolled back at the end.
Note
The following example uses the sample AdventureWorks database. The connection string provided in the sample code assumes that the database is installed and available on the local computer. Modify the connection string as necessary for your environment.
[!code-csharpSqlClient_AsyncProgramming_MultipleCommandsEx#1]