Question

J'ai passé l'après-midi à essayer de comprendre comment traduire la requête suivante en LINQ, mais je ne parviens pas à y arriver.

declare @productId int; set @productId = 3212;

select * from InformationData data where productId = @productId and orgId = 1
and exists(
    select id from (
        select coalesce(id1.id, id2.id, id3.id) as id from (
            select productId,attributeId from InformationData where productId = @productId group by productId,attributeId
        ) id
        left outer join InformationData id1 on id1.productId = id.productId and id1.attributeId = id.attributeId and id1.language = 1
        left outer join InformationData id2 on id2.productId = id.productId and id2.attributeId = id.attributeId and id2.language = 2
        left outer join InformationData id3 on id3.productId = id.productId and id3.attributeId = id.attributeId and id3.language = 0
    ) row
    where row.id = data.id
)

Le but de la requête est d'extraire les données d'une table en utilisant 2 langues de secours. Par conséquent, si les données n'existent pas dans la langue 1, elles sont extraites dans la langue 2 et si 2 n'existe pas, elles sont extraites pour la langue 0. qui est une traduction globale.

Je peux obtenir la requête interne la plupart du temps correcte (à l'exception de id1.language = 1, je n'arrive pas à la faire rejoindre un membre de la table à laquelle je participe, des idées?)

Ceci est mon code (code LINQPad):

(
    from data in (
        from d in InformationData where d.ProductId == 3212 group d by new { d.ProductId, d.AttributeId } into p select new { ProductId = p.Key.ProductId, AttributeId = p.Key.AttributeId }
    )
    join x1 in InformationData on new { a = data.ProductId, b = data.AttributeId } equals new { a = x1.ProductId, b = x1.AttributeId } into f1
        from r1 in f1.DefaultIfEmpty()
        where r1.Language == 1
    join x2 in InformationData on new { a = data.ProductId, b = data.AttributeId } equals new { a = x2.ProductId, b = x2.AttributeId } into f2
        from r2 in f2.DefaultIfEmpty()
        where r2.Language == 2
    join x3 in InformationData on new { a = data.ProductId, b = data.AttributeId } equals new { a = x3.ProductId, b = x3.AttributeId } into f3
        from r3 in f3.DefaultIfEmpty()
        where r3.Language == 2
    select new { Id = ((int?)r1.Id) ?? ((int?)r2.Id) ?? r3.Id }
).Dump();

Qui génère le code SQL suivant:

-- Region Parameters
DECLARE @p0 Int SET @p0 = 3212
DECLARE @p1 Int SET @p1 = 2
DECLARE @p2 Int SET @p2 = 2
DECLARE @p3 Int SET @p3 = 1
-- EndRegion
SELECT COALESCE([t2].[id],COALESCE([t3].[id],[t4].[id])) AS [Id]
FROM (
    SELECT [t0].[productId], [t0].[attributeId]
    FROM [InformationData] AS [t0]
    WHERE [t0].[productId] = @p0
    GROUP BY [t0].[productId], [t0].[attributeId]
    ) AS [t1]
LEFT OUTER JOIN [InformationData] AS [t2] ON ([t1].[productId] = [t2].[productId]) AND ([t1].[attributeId] = [t2].[attributeId])
LEFT OUTER JOIN [InformationData] AS [t3] ON ([t1].[productId] = [t3].[productId]) AND ([t1].[attributeId] = [t3].[attributeId])
LEFT OUTER JOIN [InformationData] AS [t4] ON ([t1].[productId] = [t4].[productId]) AND ([t1].[attributeId] = [t4].[attributeId])
WHERE ([t4].[language] = @p1) AND ([t3].[language] = @p2) AND ([t2].[language] = @p3)

Mais je ne peux pas concilier cela avec le reste de la requête. Peut-être que je suis juste fatigué. Je continue à le faire faire beaucoup de CROSS APPLY. Quelqu'un a-t-il des suggestions?

Était-ce utile?

La solution

Eh bien, après une bonne nuit de sommeil et certaines choses croustillantes, je suis devenu un peu plus brillant et j'ai trouvé la solution :) Pour ceux qui sont curieux, la voici


(
    from i in InformationData
    where (
        from data in (
            from d in InformationData where d.ProductId == 3212 group d by new { d.ProductId, d.AttributeId } into p select new { ProductId = p.Key.ProductId, AttributeId = p.Key.AttributeId }
        )
        join x1 in InformationData on new { a = data.ProductId, b = data.AttributeId, c = 1} equals new { a = x1.ProductId, b = x1.AttributeId, c = x1.Language } into f1
            from r1 in f1.DefaultIfEmpty()
        join x2 in InformationData on new { a = data.ProductId, b = data.AttributeId, c = 2 } equals new { a = x2.ProductId, b = x2.AttributeId, c = x2.Language } into f2
            from r2 in f2.DefaultIfEmpty()
        join x3 in InformationData on new { a = data.ProductId, b = data.AttributeId, c = 0 } equals new { a = x3.ProductId, b = x3.AttributeId, c = x3.Language } into f3
            from r3 in f3.DefaultIfEmpty()
        select new { Id = ((int?)r1.Id) ?? ((int?)r2.Id) ?? r3.Id }
    ).Any(d => d.Id == i.Id)
    select i
).Dump();

Et voici le code SQL généré


-- Region Parameters
DECLARE @p0 Int SET @p0 = 3212
DECLARE @p1 Int SET @p1 = 1
DECLARE @p2 Int SET @p2 = 2
DECLARE @p3 Int SET @p3 = 0
-- EndRegion
SELECT [t0].[id] AS [Id], [t0].[attributeId] AS [AttributeId], [t0].[productId] AS [ProductId], [t0].[value] AS [Value], [t0].[orgId] AS [OrgId], [t0].[version] AS [Version], [t0].[language] AS [Language], [t0].[metaType] AS [MetaType], [t0].[overload] AS [Overload], [t0].[parentId] AS [ParentId]
FROM [InformationData] AS [t0]
WHERE EXISTS(
    SELECT NULL AS [EMPTY]
    FROM (
        SELECT COALESCE([t3].[id],COALESCE([t4].[id],[t5].[id])) AS [value]
        FROM (
            SELECT [t1].[productId], [t1].[attributeId]
            FROM [InformationData] AS [t1]
            WHERE [t1].[productId] = @p0
            GROUP BY [t1].[productId], [t1].[attributeId]
            ) AS [t2]
        LEFT OUTER JOIN [InformationData] AS [t3] ON ([t2].[productId] = [t3].[productId]) AND ([t2].[attributeId] = [t3].[attributeId]) AND (@p1 = [t3].[language])
        LEFT OUTER JOIN [InformationData] AS [t4] ON ([t2].[productId] = [t4].[productId]) AND ([t2].[attributeId] = [t4].[attributeId]) AND (@p2 = [t4].[language])
        LEFT OUTER JOIN [InformationData] AS [t5] ON ([t2].[productId] = [t5].[productId]) AND ([t2].[attributeId] = [t5].[attributeId]) AND (@p3 = [t5].[language])
        ) AS [t6]
    WHERE [t6].[value] = [t0].[id]
    )

Autres conseils

En vous basant principalement sur votre description de ce que la requête est censée faire, je pense que vous pourrez peut-être obtenir le même résultat avec le "bon de commande" approprié. clause et ne récupérant que le premier résultat. Comme ceci:

var result =
(
    from d in InformationData
    where d.ProductId == 3212
    orderby ((d.language == 0) ? Int32.MaxValue : d.language)
    select d
).First();

MODIFIER : vous pouvez contrôler la priorité de la recherche en étendant la clause orderby. Par exemple, si la priorité doit être 2, puis 3, puis 1 et toute autre chose, vous pouvez le faire:

var result =
(
    from d in InformationData
    where d.ProductId == 3212
    orderby
        (d.language == 2) ? 0
        : (d.language == 3) ? 1
        : (d.language == 1) ? 2
        : Int32.MaxValue
    select d
).First();
Licencié sous: CC-BY-SA avec attribution
Non affilié à StackOverflow
scroll top