Почему чтение Dapper IEnumerable (dynamic) на порядок медленнее, чем чтение IEnumerable(IDataRecord)?
При сравнении Dapper с Enterprise Library Data Access блок доступа для получения данных через хранимую процедуру. Я вижу общее преимущество производительности около 40% при использовании Dapper, что несколько удивительно.
Однако, когда сравнение итерации и получение данных от интерфейса IEnumerable(IDataRecord) и интерфейс IEnumerable(динамическая), тип IEnumerable(IDataRecord) примерно на порядок быстрее. Это поведение хорошо понимается и должно быть ожидаемым или что-то не так - здесь?
Результаты:
IEnumerable (IDataRecord)
IEnumerable (dynamic) - использование dapperObject.параметр propertyName
Теперь самое интересное, что при использовании dapperObject["propertyName"] производительность находится на одном уровне с IDataRecord. Совсем не то, чего я ожидал.
Соответствующая часть кода профилирования
using System;
using System.Collections.Generic;
using System.Linq;
using Dapper.DataAccess;
using System.Data;
using tophat;
namespace Dapper.TestRunner
{
class Program
{
static void Main(string[] args)
{
var connectionString = "data source=WEBDBdev3,1866; User id=hsbmhw;Password=gEner4Y&M;Persist Security Info='true'; initial catalog=myhomeworks;";
//The following uses Tophat to create a singleton connection instance.
Database.Install<SqlServerConnectionFactory>(connectionString, ConnectionScope.ByRequest);
DapperTest();
DapperTest2();
EnterpriseLibraryIDataRecordTest();
}
private static void DapperTest()
{
for (int i = 0; i < 100; i++)
{
IEnumerable<dynamic> users = MyRepository.GetUsersDapper();
PopulateBusinessObjectsDynamic(users);
}
}
private static void DapperTest2()
{
for (int i = 0; i < 100; i++)
{
IEnumerable<dynamic> users = MyRepository.GetUsersDapper();
PopulateBusinessObjectsDynamic2(users);
}
}
private static void EnterpriseLibraryIDataRecordTest()
{
for (int i = 0; i < 100; i++)
{
IEnumerable<IDataRecord> users = MyRepository.GetUsersEntlib();
PopulateBusinessObjectsIDataRecord(users);
}
}
private static void PopulateBusinessObjectsDynamic(IEnumerable<dynamic> users)
{
foreach (var user in users)
{
BusinessObject bo = new BusinessObject(user);
}
}
private static void PopulateBusinessObjectsDynamic2(IEnumerable<dynamic> users)
{
foreach (var user in users)
{
BusinessObject bo = new BusinessObject(user);
}
}
private static void PopulateBusinessObjectsIDataRecord(IEnumerable<IDataRecord> users)
{
foreach (var user in users)
{
BusinessObject bo = new BusinessObject(user);
}
}
}
public class BusinessObject
{
public DateTime CreateDate { get; set; }
public String CreateDateString { get; set; }
public String FirstName { get; set; }
public bool IsApproved { get; set; }
public bool IsLockedOut { get; set; }
public DateTime LastActivityDate { get; set; }
public DateTime LastLoginDate { get; set; }
public String LastName {get;set;}
public String Organization{get;set;}
public int OrganizationId{get;set;}
public int PersonId{get;set;}
public String ProfileLastUpdatedBy{get;set;}
public DateTime ProfileLastUpdatedDate{get;set;}
public String RoleName{get;set;}
public long RowNumber{get;set;}
public int TotalCount{get;set;}
public Guid UserId{get;set;}
public string UserName {get;set;}
public string UserStatus{get;set;}
public BusinessObject(dynamic user)
{
CreateDate=user.CreateDate;
CreateDateString = user.CreateDateString;
FirstName = user.FirstName;
IsApproved = user.IsApproved;
IsLockedOut = user.IsLockedOut;
LastActivityDate= user.LastActivityDate;
LastLoginDate = user.LastLoginDate;
LastName = user.LastName;
Organization = user.organization;
OrganizationId=user.organization_id;
PersonId = user.party_id;
ProfileLastUpdatedBy = user.ProfileLastUpdatedBy;
ProfileLastUpdatedDate = user.ProfileLastUpdatedDate;
RoleName = user.RoleName;
RowNumber = user.RowNumber;
TotalCount = user.TotalCount;
UserId = user.UserId;
UserName= user.UserName;
UserStatus = user.UserStatus;
}
public BusinessObject(bool x, dynamic user)
{
CreateDate = user["CreateDate"];
CreateDateString = user["CreateDateString"];
FirstName = user["FirstName"];
IsApproved = user["IsApproved"];
IsLockedOut = user["IsLockedOut"];
LastActivityDate = user["LastActivityDate"];
LastLoginDate = user["LastLoginDate"];
LastName = user["LastName"];
Organization = user["organization"];
OrganizationId = user["organization_id"];
PersonId = user["party_id"];
ProfileLastUpdatedBy = user["ProfileLastUpdatedBy"];
ProfileLastUpdatedDate = user["ProfileLastUpdatedDate"];
RoleName = user["RoleName"];
RowNumber = user["RowNumber"];
TotalCount = user["TotalCount"];
UserId = user["UserId"];
UserName = user["UserName"];
UserStatus = user["UserStatus"];
}
public BusinessObject(IDataRecord user)
{
CreateDate = (DateTime)user["CreateDate"];
CreateDateString = (string)user["CreateDateString"];
FirstName = (string)user["FirstName"];
IsApproved = (bool)user["IsApproved"];
IsLockedOut = (bool)user["IsLockedOut"];
LastActivityDate = (DateTime)user["LastActivityDate"];
LastLoginDate = (DateTime)user["LastLoginDate"];
LastName = (string)user["LastName"];
Organization = (string)user["organization"];
OrganizationId = (int)user["organization_id"];
PersonId = (int)user["party_id"];
ProfileLastUpdatedBy = (string)user["ProfileLastUpdatedBy"];
ProfileLastUpdatedDate = (DateTime)user["ProfileLastUpdatedDate"];
RoleName = (string)user["RoleName"];
RowNumber = (long)user["RowNumber"];
TotalCount = (int)user["TotalCount"];
UserId = (Guid)user["UserId"];
UserName = (string)user["UserName"];
UserStatus = (string)user["UserStatus"];
}
}
}
1 ответ:
На самом деле вы, кажется, запускаете один и тот же тест дважды; вход (
users
) один и тот же, отсюда:private static void DapperTest() { for (int i = 0; i < 100; i++) { IEnumerable<dynamic> users = MyRepository.GetUsersDapper(); PopulateBusinessObjectsDynamic(users); } } private static void DapperTest2() { for (int i = 0; i < 100; i++) { IEnumerable<dynamic> users = MyRepository.GetUsersDapper(); PopulateBusinessObjectsDynamic2(users); } }
И фактическое "что мы делаем" здесь то же самое:
private static void PopulateBusinessObjectsDynamic(IEnumerable<dynamic> users) { foreach (var user in users) { BusinessObject bo = new BusinessObject(user); } } private static void PopulateBusinessObjectsDynamic2(IEnumerable<dynamic> users) { foreach (var user in users) { BusinessObject bo = new BusinessObject(user); } }
Итак... Я должен сделать вывод: "комбинация JIT, кэширования данных (на сервере базы данных), загрузки сборок/проверки/слияния, пула соединений и функций кэша
dynamic
сделала второй тест более быстрым".Обратите внимание, что сторона
dynamic
dapper в любом случае предназначена только для специального использования. Если ты если бы вы хотели оптимальное лицо dapper, вы бы использовалиQuery<T>
, а неdynamic
.В частности, насколько мне известно, не строить из щеголеватого поддерживает строковый индексатор на
dynamic
API-интерфейс. Объект реализуетIDictionary<string,object>
для доступа к члену, но вам нужно будет явно привести к этому, чтобы использовать его - вы не можете сделатьuser["PropName"]
, Еслиuser
типизируется какdynamic
(Если я ошибаюсь, пожалуйста, скажите мне!).Как это происходит, неизданный код" git "(для API
dynamic
) заметно быстрее, чем текущий " nuget" реализация-но это немного касательная к этому конкретному вопросу.