You can use Dapper to access a database and write queries in a more object-oriented manner. For example, it allows you to execute SQL commands and map the resulting data into a list of objects. With Dapper, you can easily select data from a database and project the result into a list of objects.
Dapper provides several methods for selecting data, depending on how you want to work with the data you retrieve.
Method | Description |
---|---|
Query |
Returns an enumerable of dynamic types |
Query<T> |
Returns an enumerable of the type specified by the T parameter |
QueryAsync |
Returns an enumerable of dynamic types asynchronously |
QueryAsync<T> |
Returns an enumerable of the type specified by the T parameter asynchronously |
To select data from a table or view using Dapper, you need to create an instance of the IDbConnection
interface. Then you can use the Query<T>()
or QueryAsync<T>
method to execute the SELECT
query and store the results in a list of objects.
Dapper Query
The primary methods used for selecting data are Query
and Query<T>
and their async
counterparts. The difference between Query
and Query<T>
is that Query
returns results as an IEnumerable<dynamic>
and the Query<T>
returns an IEnumerable
of the type specified by the type argument. The following examples query the Customers table from the ubiquitous Microsoft Northwind sample database. The schema of the table is as follows:
For example, suppose you have a database table called Customers. You could use the following code to select all rows from the Customers table and project them into a dynamic list of objects:
using (var connection = new SqlConnection(connectionString))
{
var sql = "SELECT * FROM Customers";
var customers = connection.Query(sql);
foreach(var customer in customers)
{
Console.WriteLine($"{customer.CustomerID} {customer.CompanyName}");
}
}
Note that there is no explicit call to open the connection in the code above. Instead, Dapper will check the state of the connection and open it if it is closed.
Working with dynamic types should ordinarily be avoided because the risk of introducing run-time errors is high. It can be easy to mistype a property name, especially as they are case-sensitive. For example, if you referred to the key column as CustomerId
instead of CustomerID
, there would be no help from Intellisense or exceptions raised. Nothing would be output to the console window at all. Therefore the recommendation is to use the Query<T>
methods that takes a type parameter representing the data type that the query returns.
The next example works with a Customer
type, which, following the table schema, has the following definition:
public class Customer
{
public string CustomerID { get; set; }
public string CompanyName { get; set; }
public string ContactName { get; set; }
public string ContactTitle { get; set; }
public string Address { get; set; }
public string City { get; set; }
public string Region { get; set; }
public string PostalCode { get; set; }
public string Country { get; set; }
public string Phone { get; set; }
public string Fax { get; set; }
}
The following example is identical to the first one except for the presence of the <Customer>
type parameter:
using (var connection = new SqlConnection(connectionString))
{
var sql = "SELECT * FROM Customers";
var customers = connection.Query<Customer>(sql);
foreach(var customer in customers)
{
Console.WriteLine($"{customer.CustomerID} {customer.CompanyName}");
}
}
Dapper QueryAsync
To execute the query asynchronously, Dapper provides the QueryAsync
method, an asynchronous version of the Query
method.
using (var connection = new SqlConnection(connectionString))
{
var sql = "SELECT * FROM Customers";
var customers = await connection.QueryAsync(sql);
foreach(var customer in customers)
{
Console.WriteLine($"{customer.CustomerID} {customer.CompanyName}");
}
}
The asynchronous version of the Query<T>
method is QueryAsync<T>
:
using (var connection = new SqlConnection(connectionString))
{
var sql = "SELECT * FROM Customers";
var customers = await connection.QueryAsync<Customer>(sql);
foreach(var customer in customers)
{
Console.WriteLine($"{customer.CustomerID} {customer.CompanyName}");
}
}
Note if you want to use QueryAsync
with buffered = false
, you should use instead the QueryUnbufferedAsync method.