Как мне сопоставить списки вложенных объектов с помощью Dapper - PullRequest
101 голосов
/ 22 сентября 2011

В настоящее время я использую Entity Framework для доступа к БД, но хочу взглянуть на Dapper. У меня есть такие классы:

public class Course{
   public string Title{get;set;}
   public IList<Location> Locations {get;set;}
   ...
}

public class Location{
   public string Name {get;set;}
   ...
}

Таким образом, один курс можно преподавать в нескольких местах. Entity Framework выполняет сопоставление для меня, поэтому мой объект Course заполняется списком местоположений. Как мне поступить с Dapper, это вообще возможно или мне нужно сделать это за несколько шагов?

Ответы [ 7 ]

148 голосов
/ 19 июля 2013

Кроме того, вы можете использовать один запрос с поиском:

var lookup = new Dictionary<int, Course>();
conn.Query<Course, Location, Course>(@"
    SELECT c.*, l.*
    FROM Course c
    INNER JOIN Location l ON c.LocationId = l.Id                    
    ", (c, l) => {
        Course course;
        if (!lookup.TryGetValue(c.Id, out course))
            lookup.Add(c.Id, course = c);
        if (course.Locations == null) 
            course.Locations = new List<Location>();
        course.Locations.Add(l); /* Add locations to course */
        return course;
     }).AsQueryable();
var resultList = lookup.Values;

Смотрите здесь https://www.tritac.com/blog/dappernet-by-example/

45 голосов
/ 23 сентября 2011

Dapper - не полноценный ORM, он не обрабатывает магические запросы и тому подобное.

Для вашего конкретного примера, вероятно, сработает следующее:

Выберите курсы:

var courses = cnn.Query<Course>("select * from Courses where Category = 1 Order by CreationDate");

Выберите соответствующее отображение:

var mappings = cnn.Query<CourseLocation>(
   "select * from CourseLocations where CourseId in @Ids", 
    new {Ids = courses.Select(c => c.Id).Distinct()});

Выберитесоответствующие местоположения

var locations = cnn.Query<Location>(
   "select * from Locations where Id in @Ids",
   new {Ids = mappings.Select(m => m.LocationId).Distinct()}
);

Составьте карту всего этого

Оставляя это на усмотрение читателя, вы создаете несколько карт и перебираете свои курсы с указанием местоположений.

Предупреждение трюк in сработает, если у вас меньше 2100 поисков (Sql Server), если у вас больше, вы, вероятно, захотите изменить запрос наselect * from CourseLocations where CourseId in (select Id from Courses ... ) если это так, вы можете также вытащить все результаты за один раз, используя QueryMultiple

29 голосов
/ 17 октября 2017

Нет необходимости lookup Словарь

var coursesWithLocations = 
    conn.Query<Course, Location, Course>(@"
        SELECT c.*, l.*
        FROM Course c
        INNER JOIN Location l ON c.LocationId = l.Id                    
        ", (course, location) => {
            course.Locations = course.Locations ?? new List<Location>();
            course.Locations.Add(location); 
            return course;
        }).AsQueryable();
26 голосов
/ 23 февраля 2017

Я знаю, что очень опаздываю, но есть и другой вариант. Вы можете использовать QueryMultiple здесь. Примерно так:

var results = cnn.QueryMultiple(@"
    SELECT * 
      FROM Courses 
     WHERE Category = 1 
  ORDER BY CreationDate
          ; 
    SELECT A.*
          ,B.CourseId 
      FROM Locations A 
INNER JOIN CourseLocations B 
        ON A.LocationId = B.LocationId 
INNER JOIN Course C 
        ON B.CourseId = B.CourseId 
       AND C.Category = 1
");

var courses = results.Read<Course>();
var locations = results.Read<Location>(); //(Location will have that extra CourseId on it for the next part)
foreach (var course in courses) {
   course.Locations = locations.Where(a => a.CourseId == course.CourseId).ToList();
}
2 голосов
/ 10 октября 2018

Извините, что опоздал на вечеринку (как всегда).Для меня проще использовать Dictionary, , как Jeroen K сделал , с точки зрения производительности и читабельности.Кроме того, чтобы избежать умножения заголовка в местах , я использую Distinct() для удаления потенциальных дублирований:

string query = @"SELECT c.*, l.*
    FROM Course c
    INNER JOIN Location l ON c.LocationId = l.Id";
using (SqlConnection conn = DB.getConnection())
{
    conn.Open();
    var courseDictionary = new Dictionary<Guid, Course>();
    var list = conn.Query<Course, Location, Course>(
        query,
        (course, location) =>
        {
            if (!courseDictionary.TryGetValue(course.Id, out Course courseEntry))
            {
                courseEntry = course;
                courseEntry.Locations = courseEntry.Locations ?? new List<Location>();
                courseDictionary.Add(courseEntry.Id, courseEntry);
            }

            courseEntry.Locations.Add(location);
            return courseEntry;
        },
        splitOn: "Id")
    .Distinct()
    .ToList();

    return list;
}
2 голосов
/ 14 февраля 2014

Чего-то не хватает. Если вы не укажете каждое поле из Locations в запросе SQL, объект Location не может быть заполнен. Взгляните:

var lookup = new Dictionary<int, Course>()
conn.Query<Course, Location, Course>(@"
    SELECT c.*, l.Name, l.otherField, l.secondField
    FROM Course c
    INNER JOIN Location l ON c.LocationId = l.Id                    
    ", (c, l) => {
        Course course;
        if (!lookup.TryGetValue(c.Id, out course)) {
            lookup.Add(c.Id, course = c);
        }
        if (course.Locations == null) 
            course.Locations = new List<Location>();
        course.Locations.Add(a);
        return course;
     },
     ).AsQueryable();
var resultList = lookup.Values;

Используя l.* в запросе, у меня был список местоположений, но без данных.

0 голосов
/ 18 июля 2017

Не уверен, что кому-то это нужно, но у меня есть динамическая версия без модели для быстрого и гибкого кодирования.

var lookup = new Dictionary<int, dynamic>();
conn.Query<dynamic, dynamic, dynamic>(@"
    SELECT A.*, B.*
    FROM Client A
    INNER JOIN Instance B ON A.ClientID = B.ClientID                
    ", (A, B) => {
        // If dict has no key, allocate new obj
        // with another level of array
        if (!lookup.ContainsKey(A.ClientID)) {
            lookup[A.ClientID] = new {
                ClientID = A.ClientID,
                ClientName = A.Name,                                        
                Instances = new List<dynamic>()
            };
        }

        // Add each instance                                
        lookup[A.ClientID].Instances.Add(new {
            InstanceName = B.Name,
            BaseURL = B.BaseURL,
            WebAppPath = B.WebAppPath
        });

        return lookup[A.ClientID];
    }, splitOn: "ClientID,InstanceID").AsQueryable();

var resultList = lookup.Values;
return resultList;
...