Dapper комплексное картирование Dapper.Extensions Dapper.FluentMap
У меня есть небольшая проблема с моим кодом, и я стараюсь играть с ним красиво.
Когда я говорю, мой код был унаследован, так что это не мой дизайн.
Я пытаюсь заменить Entity Framework, так как обращения к базе данных менее эффективны, поэтому я хотел лучше контролировать SQL, поэтому Dapper показался мне очевидным выбором.
Проблема, с которой я столкнулся, заключается в том, что я пытаюсь сопоставить классы poco, которые у меня есть, с более сложным мультизапросом.
Вопрос у меня заключается в следующем:
public Feed GetFeedDapper(int feedId)
{
Feed feed = null;
var sql =
@"
SELECT * FROM Feeds WHERE FeedId= @FeedId
SELECT * FROM FeedFilterParameters WHERE FeedId = @FeedId
SELECT * FROM TeamFeeds WHERE FeedId = @FeedId";
using (var multi = DbConnection.QueryMultiple(sql, new { FeedId = feedId }))
{
feed = multi.Read<Feed>().Single();
feed.Parameters = multi.Read<FeedFilterParameter>().ToList();
feed.TeamFeeds = multi.Read<TeamFeed>().ToList();
}
return feed;
}
Это дает мне правильные результаты из базы данных, и это нормально, проблема в том, что не все свойства объекта Feed сопоставляются. Фид имеет свойство InboundProperties типа InboundProperties, как показано ниже, и в базе данных они хранятся как InboundProperties_{PropName}. Эти свойства не отображаются, и все, что я пробую в DapperExtensions или FluentMap, работает.
public class InboundProperties
{
public string ExternalRef { get; set; }
public string ExternalRefPrevious { get; set; }
public string ExternalId { get; set; }
public string ExternalName { get; set; }
public string ExternalToken { get; set; }
public int ExternalAPICounts { get; set; }
public string ExternalLink { get; set; }
public string ExternalPicture { get; set; }
public string LastProcessedMessageId { get; set; }
public long? LastProcessedMessageTime { get; set; }
public DateTime? MessageCountStartDT { get; set; }
public Int32 TenancyId { get; set; }
public virtual int FeedScopeInt { get; set; }
}
Может кто-нибудь помочь мне сопоставить эти свойства??
3 ответа
Ты пробовал что-то подобное:
Feed feed = null;
var sql =
@"
SELECT * FROM Feeds WHERE FeedId= @FeedId
SELECT InboundProperties_ExternalRef as ExternalRef, InboundProperties_ExternalRefPrevious as ExternalRefPrevious FROM Feeds as InboundProperties WHERE FeedId= @FeedId
SELECT * FROM FeedFilterParameters WHERE FeedId = @FeedId
SELECT * FROM TeamFeeds WHERE FeedId = @FeedId";
using (var multi = DbConnection.QueryMultiple(sql, new { FeedId = feedId }))
{
feed = multi.Read<Feed>().Single();
feed.InboundProperties = multi.Read<InboundProperties>().Single();
feed.Parameters = multi.Read<FeedFilterParameter>().ToList();
feed.TeamFeeds = multi.Read<TeamFeed>().ToList();
}
return feed;
Я сопоставил только 2 первых свойства, но если это сработает, вы будете знать, как отобразить все свойства.
Вот как я это решил, но принял ответ #rraszewski, так как он тоже работает, но просто означает, что мне нужно позаботиться обо всех выборках очень ручным способом.
public Feed GetFeedDapper(int feedId)
{
Feed feed = null;
var multiPredicate = new GetMultiplePredicate();
multiPredicate.Add<Feed>(Predicates.Field<Feed>(x => x.FeedId, Operator.Eq, feedId));
multiPredicate.Add<InboundProperties>(Predicates.Field<InboundProperties>(x => x.FeedId, Operator.Eq, feedId));
multiPredicate.Add<OutboundProperties>(Predicates.Field<OutboundProperties>(x => x.FeedId, Operator.Eq, feedId));
multiPredicate.Add<FeedFilterParameter>(Predicates.Field<FeedFilterParameter>(x => x.FeedId, Operator.Eq, feedId));
multiPredicate.Add<TeamFeed>(Predicates.Field<TeamFeed>(x => x.FeedId, Operator.Eq, feedId));
var result = DbConnection.GetMultiple(multiPredicate);
feed = result.Read<Feed>().Single();
feed.InboundProperties = result.Read<InboundProperties>().Single();
feed.OutboundProperties = result.Read<OutboundProperties>().Single();
feed.Parameters = result.Read<FeedFilterParameter>().ToList();
feed.TeamFeeds = result.Read<TeamFeed>().ToList();
return feed;
}
Затем я сопоставляю классы:
public class FeedMapper : ClassMapper<Feed>
{
public FeedMapper()
{
base.Table("Feeds");
Map(f => f.FeedId).Key(KeyType.Identity);
Map(f => f.Owner).Ignore();
Map(f => f.TeamFeeds).Ignore();
Map(f => f.FeedDirection).Ignore();
Map(f => f.InboundProperties).Ignore();
Map(f => f.Parameters).Ignore();
Map(f => f.OutboundProperties).Ignore();
Map(f => f.RelatedTeams).Ignore();
AutoMap();
}
}
public class InboundPropertiesMapper : ClassMapper<InboundProperties>
{
public InboundPropertiesMapper()
{
base.Table("Feeds");
Map(f => f.FeedId).Key(KeyType.Identity);
Map(f => f.Channel).Ignore();
Map(f => f.FeedScope).Ignore();
Map(f => f.ChannelInt).Column("InboundProperties_ChannelInt");
Map(f => f.ExternalAPICounts).Column("InboundProperties_ExternalAPICounts");
Map(f => f.ExternalId).Column("InboundProperties_ExternalId");
Map(f => f.ExternalLink).Column("InboundProperties_ExternalLink");
Map(f => f.ExternalName).Column("InboundProperties_ExternalName");
Map(f => f.ExternalPicture).Column("InboundProperties_ExternalPicture");
Map(f => f.ExternalRef).Column("InboundProperties_ExternalRef");
Map(f => f.ExternalRefPrevious).Column("InboundProperties_ExternalRefPrevious");
Map(f => f.ExternalToken).Column("InboundProperties_ExternalToken");
Map(f => f.FeedScopeInt).Column("InboundProperties_FeedScopeInt");
Map(f => f.LastProcessedMessageId).Column("InboundProperties_LastProcessedMessageId");
Map(f => f.LastProcessedMessageTime).Column("InboundProperties_LastProcessedMessageTime");
Map(f => f.MessageCountStartDT).Column("InboundProperties_MessageCountStartDT");
Map(f => f.TenancyId).Column("InboundProperties_TenancyId");
AutoMap();
}
}
public class OutboundPropertiesMapper : ClassMapper<OutboundProperties>
{
public OutboundPropertiesMapper()
{
base.Table("Feeds");
Map(f => f.FeedId).Key(KeyType.Identity);
Map(f => f.Channel).Ignore();
Map(f => f.FeedType).Ignore();
Map(f => f.OutboundFeedTypeInt).Column("OutboundProperties_OutboundFeedTypeInt");
Map(f => f.TenancyId).Column("OutboundProperties_TenancyId");
AutoMap();
}
}
Я не был удовлетворен представленными здесь решениями (слишком подверженными ошибкам), поэтому я решил создать метод расширения для отображения объектов с помощью отражения.
public static class ConnectionExtensions
{
public static IEnumerable<T> QueryIncludeNestedObjects<T>(this SqlConnection connection, string sql)
{
var queryResults = connection.Query<dynamic>(sql);
var typeOfTMain = typeof(T);
foreach(var row in queryResults)
{
var mappedObject = Activator.CreateInstance<T>();
foreach (var col in row)
{
var colKey = (string)col.Key;
var colValue = (object)col.Value;
if(colKey.Contains("_"))
{
var subObjNameAndProp = colKey.Split('_');
var subProperty = typeOfTMain.GetProperty(subObjNameAndProp[0]);
if (subProperty == null) continue;
var subObj = subProperty.GetValue(mappedObject);
if(subObj == null)
{
subObj = Activator.CreateInstance(subProperty.PropertyType);
typeOfTMain.GetProperty(subObjNameAndProp[0]).SetValue(mappedObject, subObj);
}
subObj.GetType().GetProperty(subObjNameAndProp[1])
.SetValue(subObj, colValue);
}
else
typeOfTMain.GetProperty(colKey)?.SetValue(mappedObject, colValue);
}
yield return mappedObject;
}
}
}
Тогда вы используете это так:
dbConnection.QueryIncludeNestedObjects<Feed>("SELECT * FROM Feeds");
Если вы хотите использовать его внутри QueryMultiple, как в этом вопросе, просто измените метод следующим образом:
public static IEnumerable<T> ReadIncludeNestedObjects<T>(this GridReader gridReader)
{
var queryResults = gridReader.Read<dynamic>();
...