Проблемы производительности с запросом LINQ при включении предложения OR - PullRequest
0 голосов
/ 05 декабря 2011

Следующий запрос LINQ to Entities существенно замедляется, когда в запрос включена строка комментария ниже. Есть ли лучший способ выразить это?

Предложение 'OR' должно учитывать только следующие строки:

((o.Type_ID == (int) RecordEnums.RecordType.Lead) && (o.Item_ID == l1.Lead_ID)) 
|| ((o.Type_ID == (int)RecordEnums.RecordType.Opportunity)(o.Item_ID == o1.Opportunity_ID)) 

Полный запрос

return withItemsPending
                       ? (from l1 in db.Leads
                          from o1 in db.Opportunities.Where(x => (x.Lead_ID == l1.Lead_ID) && (x.Company_ID == companyId)).DefaultIfEmpty()
                          from l2
                              in
                              db.Tasks.Where(
                                  o =>
                                  ((o.IsCompleted ?? false) == false) &&
                                  (o.TaskType_ID == typeId) &&
                                  ((o.Type_ID == (int) RecordEnums.RecordType.Lead) && (o.Item_ID == l1.Lead_ID)) 
                                  //|| ((o.Type_ID == (int)RecordEnums.RecordType.Opportunity) && (o.Item_ID == o1.Opportunity_ID)) 
                                  &&
                                  (o.Due_Date > EntityFunctions.AddDays(DateTime.Now, -1)))


                          where (l1.Company_ID == companyId)
                          select l1)


                       : (from l1 in db.Leads where (0 == 1) select l1);
        }

Вот неправильный запрос:

SELECT     Extent1.Lead_ID
FROM         Leads AS Extent1 LEFT OUTER JOIN
                      Opportunities AS Extent2 ON Extent2.Lead_ID = Extent1.Lead_ID AND Extent2.Company_ID = 118 INNER JOIN
                      Tasks AS Extent3 ON 0 = (CASE WHEN ([Extent3].[IsCompleted] IS NULL) THEN CAST(0 AS bit) ELSE [Extent3].[IsCompleted] END) AND Extent3.TaskType_ID = 1 AND 
                      5 = Extent3.Type_ID AND Extent3.Item_ID = Extent1.Lead_ID OR
                      4 = Extent3.Type_ID AND Extent3.Item_ID = Extent2.Opportunity_ID AND Extent3.Due_Date > DATEADD(day, - 1, SysDateTime())
WHERE     (Extent1.Company_ID = 118)

1 Ответ

1 голос
/ 05 декабря 2011

Если вы хотите «дать мне элементы, которые не были завершены и должны быть оплачены позднее, чем вчера, и являются ЛИБО ИЛИ ВОЗМОЖНОСТЬЮ», вам необходимо добавить дополнительные скобки вокруг 2 операторов, которые вы хотите ИЛИ. В настоящее время вы говорите: «Дайте мне предметы, которые либо не были завершены, ИЛИ ИЛИ ИЛИ ВОЗМОЖНОСТЬ И СРОК ДЕЙСТВИЯ ПОСЛЕ ВРЕМЕНИ».

Код будет таким:

return withItemsPending
                       ? (from l1 in db.Leads
                          from o1 in db.Opportunities.Where(x => (x.Lead_ID == l1.Lead_ID) && (x.Company_ID == companyId)).DefaultIfEmpty()
                          from l2
                              in
                              db.Tasks.Where(
                                  o =>
                                  ((o.IsCompleted ?? false) == false) &&
                                  (o.TaskType_ID == typeId) &&
                                  (((o.Type_ID == (int) RecordEnums.RecordType.Lead) && (o.Item_ID == l1.Lead_ID)) 
                                  || ((o.Type_ID == (int)RecordEnums.RecordType.Opportunity) && (o.Item_ID == o1.Opportunity_ID)))
                                  &&
                                  (o.Due_Date > EntityFunctions.AddDays(DateTime.Now, -1)))


                          where (l1.Company_ID == companyId)
                          select l1)


                       : (from l1 in db.Leads where (0 == 1) select l1);
        }
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...