SQL для LINQ помогает вычитать данные из одной таблицы из другой таблицы (несколько строк)

У меня есть две таблицы, которые я использую для определения целевого показателя «Часы» и «Завершенные часы». одна таблица содержит исходные значения, а другая — часы. Журналов может быть несколько. Я пытаюсь создать запрос linq, который добавляет зарегистрированные часы для указанного пользователя и вычитает это число из основной таблицы. например:

Основная таблица

ID:1 Пользователь:Пользователь1 Цель:20

Таблица журналов

ID:1 Пользователь:Пользователь1 Выполнено:5

ID:2 Пользователь:Пользователь1 Выполнено:8

И возвращаемое значение должно быть

ID:1 Пользователь:Пользователь1 Цель:20 Завершено:13 Осталось: 7

вот мой linq-запрос

var ah = db.AccumulatedHours.ToList();
var gh = db.GoalHours.Where(m=>m.UserID == 4).ToList();

var rmng = from GH in gh
join AH in ah
on GH.ID equals AH.GH_ID
Select new GoalHours
{
ID = GH.ID,
GoalHours = GH.Target
Completed = AH.Completed
Remaining = GH.Target - AH.Completed
};

«но я не могу заставить его работать в LINQ». Что вы пробовали и что не сработало должным образом?

David 23.08.2024 21:31

@Дэвид, я только что обновил вопрос своим запросом linq

Manny Sanchez 23.08.2024 21:42

Пожалуйста, добавьте тег для библиотеки данных, которую вы используете, я думаю, [entity-framework-core] + тег для версии [ef-core-xx]. Кроме того, было бы полезно увидеть определения классов. Детали имеют значение.

Gert Arnold 23.08.2024 21:43

Если связь между GoalHours и AccumulatedHours один-ко-многим, вам потребуется сгруппировать по значениям GoalHours.ID and GoalHours.Target, and then sum the AccumulatedHours.Completed`. Если для данной строки AccumulatedHours может быть ноль строк GoalHours, вам понадобится эквивалент LEFT JOIN в выражении LINQ — возможно, с использованием .GroupJoin().

T N 24.08.2024 04:30

Какую базу данных вы используете? (SQL-сервер? MySQL?) Какая объектная модель? (Entity Framework? EF Core?) Какие версии каждого?

T N 24.08.2024 15:44
Стоит ли изучать PHP в 2026-2027 годах?
Стоит ли изучать PHP в 2026-2027 годах?
Привет всем, сегодня я хочу высказать свои соображения по поводу вопроса, который я уже много раз получал в своем сообществе: "Стоит ли изучать PHP в...
Поведение ключевого слова "this" в стрелочной функции в сравнении с нормальной функцией
Поведение ключевого слова "this" в стрелочной функции в сравнении с нормальной функцией
В JavaScript одним из самых запутанных понятий является поведение ключевого слова "this" в стрелочной и обычной функциях.
Приемы CSS-макетирования - floats и Flexbox
Приемы CSS-макетирования - floats и Flexbox
Здравствуйте, друзья-студенты! Готовы совершенствовать свои навыки веб-дизайна? Сегодня в нашем путешествии мы рассмотрим приемы CSS-верстки - в...
Тестирование функциональных ngrx-эффектов в Angular 16 с помощью Jest
В системе управления состояниями ngrx, совместимой с Angular 16, появились функциональные эффекты. Это здорово и делает код определенно легче для...
Концепция локализации и ее применение в приложениях React ⚡️
Концепция локализации и ее применение в приложениях React ⚡️
Локализация - это процесс адаптации приложения к различным языкам и культурным требованиям. Это позволяет пользователям получить опыт, соответствующий...
Пользовательский скаляр GraphQL
Пользовательский скаляр GraphQL
Листовые узлы системы типов GraphQL называются скалярами. Достигнув скалярного типа, невозможно спуститься дальше по иерархии типов. Скалярный тип...
0
5
104
3
Перейти к ответу Данный вопрос помечен как решенный

Ответы 3

Предполагая, что в вашем приложении определен некоторый контекст БД, вы можете использовать:

context.GoalHours
    .Join(
        context.AccumulatedHours, 
        x => x.ID, 
        x => x.GH_ID, 
        (x, y) => new { x.ID, x.Target, y.Completed, Remaining = x.Target - y.Completed })
    .ToArray(); // or .ToArrayAsync()
Ответ принят как подходящий

Ниже я определяю два класса:

    public class AccumulatedHours
    {
        public int Id { get; set; }
        public int UserId { get; set; }
        public int Completed { get; set; }
    }

    public class GlobalHours
    {
        public int Id { get; set; }
        public int UserId { get; set; }
        public int Target { get; set; }
    }

Затем я использовал ниже Linq. Вам нужно использовать on GH.UserId equals AH.UserId into userInfos select new для присоединения:

var ah = new List<AccumulatedHours>
{
    new AccumulatedHours {Id = 1, UserId = 1, Completed = 5},
    new AccumulatedHours {Id = 2, UserId = 1, Completed = 8}
};
var gh = new List<GlobalHours>
{
    new GlobalHours {Id = 1, UserId = 1, Target = 20}
};

var joined = (from GH in gh
    join AH in ah on GH.UserId equals AH.UserId into userInfos
    select new
    {
        userId = GH.UserId,
        target = GH.Target,
        sumOfCompleted = userInfos.Sum(r => r.Completed),
        remaining = GH.Target - userInfos.Sum(r => r.Completed)
    }).ToList();

Результаты такие:

Во-первых, вам не следует обращаться к данным таблицы базы данных и выполнять .ToList() перед применением всех соединений, фильтров и селекторов. Это приведет к излишней загрузке данных (возможно, всех строк и всех столбцов) в коллекцию в памяти. После этого любые запросы, примененные к этим спискам, не будут иметь доступа к исходным ключам и индексам и будут выполнять в памяти эквивалент сканирования таблицы.

Судя по вашему утверждению «Может быть несколько журналов» и столбцам идентификаторов, указанным в вашем объединении, становится очевидным, что связь между GoalHours и AccumulatedHours — «один ко многим». Чтобы получить желаемые результаты, ваш запрос необходимо сгруппировать по GoalHours.ID и GoalHours.Target, а затем суммировать значения AccumulatedHours.Completed.

Если существует вероятность наличия GoalHours без совпадающих AccumulatedHours строк (что может быть обычным явлением, если часы еще не записаны), вам потребуется выполнить LEFT JOIN между этими таблицами или эквивалент LINQ. Вам также потребуется обработать условие нулевой суммы с помощью функции SQL ISNULL() или COALESCE() или эквивалентного оператора объединения нулей C# ??.

Предполагая, что необходимо левое соединение, я считаю, что вы можете достичь желаемых результатов с помощью чего-то вроде следующего SQL:

DECLARE @UserID INT = 4

SELECT
    gh.ID,
    gh.Target AS GoalHours,
    ISNULL(SUM(ah.Completed), 0) AS Completed,
    gh.Target - ISNULL(SUM(ah.Completed), 0) AS Remaining
FROM GoalHours gh
LEFT JOIN AccumulatedHours ah
    ON gh.ID = ah.GH_ID
WHERE gh.UserID = @UserID
GROUP BY gh.ID, gh.Target;

Левые соединения сложны в LINQ. Один из подходов — выполнить «групповое объединение» с использованием синтаксиса join ... on ... into ..., который объединяет строки с левой стороны объединения с (возможно, пустой) коллекцией строк с правой стороны объединения. Это удобно обеспечивает как левое соединение, так и эквивалент группировки. Затем необходимо выбрать, суммировать и объединить результаты.

Тогда эквивалент запроса LINQ будет выглядеть примерно так:

int userID = 4

var query =
    from gh in dbContext.GoalHours
    where gh.UserID == userID
    join ah in dbContext.AccumulatedHours
        on gh.ID equals ah.GH_ID
        into ahGroup
    select new
    {
        ID = gh.ID,
        GoalHours = gh.Target,
        Completed = ahGroup.Sum(ah => (decimal?)ah.Completed) ?? 0,
        Remaining = gh.Target - (ahGroup.Sum(ah => (decimal?)ah.Completed) ?? 0)
    };

Как указано в комментариях ниже, по крайней мере некоторые версии платформы не поддерживают подход GroupJpoin. Ниже приведено альтернативное LINQ-представление левого соединения, использующее технику .DefaultIfEmpty(). (Однако логика обработки null может нуждаться в некоторых изменениях.)

var query2 = 
    from gh in dbContext.GoalHours
    where gh.UserID == userID
    join ah in dbContext.AccumulatedHours
        on gh.ID equals ah.GH_ID
        into ahGroup
    from ahd in ahGroup.DefaultIfEmpty()
    group  new { gh, ahd } by new { gh.ID, gh.Target } into grouped
    select new
    {
        ID = grouped.Key.ID,
        GoalHours = grouped.Key.Target,
        Completed = grouped.Sum(g => (decimal?)g.ahd?.Completed) ?? 0,
        Remaining = grouped.Key.Target - (grouped.Sum(g => (decimal?)g.ahd?.Completed) ?? 0)
    };

Оператор приведения (decimal?) необходим для правильной обработки сумм для пустых списков. Функция SQL-сервера SUM() вернет значение null для пустого списка (или списка, не содержащего ненулевых значений). Использование этого приведения вместе с ?? 0 гарантирует, что сумма для пустого списка вернет ноль. Вам нужен соответствующий тип C#, соответствующий типу вашего столбца Completed.

(Вышеупомянутое не проверено. Если я допустил ошибку, я приветствую комментарии или исправления.)

В csharp это where gh.UserID == userID

rachel 24.08.2024 05:29

Если вы хотите GROUP BY, используйте LINQ group by, ваш запрос неэффективен.

Svyatoslav Danyliv 24.08.2024 06:22

@SvyatlavDanyliv - Вы знакомы с функцией .GroupJoin() и ее эквивалентным синтаксисом join ... into? Это работает иначе, чем обычное соединение. Эта скрипта .NET дает правильные результаты для LINQ-to-объектов, хотя я признаю, что не тестировал ее с LINQ-to-SQL. См. также этот ТАК ответ.

T N 24.08.2024 07:58

Я знаю все операторы LINQ и то, как они переводятся в SQL. GroupJoin не имеет аналогов в мире SQL. Ваш запрос создаст два подзапроса, что не эквивалентно группировке.

Svyatoslav Danyliv 24.08.2024 08:42

Изучая различные статьи, выяснилось, что GroupJoin() поддерживается в EF 6 и 7, но изначально не поддерживался EF Core. В другом примечании говорилось, что в более поздней версии реализован GroupJoin(), но как OUTER APPLY вместо настоящего LEFT JOIN. Кажется, что ответ зависит от объектной модели. Я обновлю ответ альтернативной версией, в которой используется .DefaultIfEmpty(), которая, похоже, имеет более широкую поддержку.

T N 24.08.2024 15:56

Другие вопросы по теме