Ir para conteúdo

Arquivado

Este tópico foi arquivado e está fechado para novas respostas.

  • 0
Didyo

JOIN limitar registros da tabela filho por cada linha da tabela pai

Pergunta

Olá pessoal, gostaria de saber se é possível e como limitar a quantidade de linha retornadas usando JOIN.
A ideia é limitar a quantidade de linhas retornadas da tabela filho em relação a cada linha da tabela pai.

Ex. Tenho 10 registros de imóveis na tabela_imv e cada um destes possui mais de 12 imagens na tabela_imgs (cada imagem em uma linha diferente).

Usando o LEFT JOIN gostaria que para cada imóvel fossem trazidas apenas 5 imagens usando LIMIT 5 de modo que a cada 5 imagens encontradas, a busca cessasse na tabela filho, e retomasse procurando as 5 primeiras imagens do próximo imóvel.

Desde já agradeço a atenção.

Compartilhar este post


Link para o post
Compartilhar em outros sites

7 respostas a esta questão

Recommended Posts

Olá pessoal, tive pensando em resolver a questão com algo tipo 

 

SELECT c.`cat_id` as id, c.`cat_nome_fantasia` as empresa, i.img_nome as imagem FROM `catalogo` c 
LEFT JOIN (SELECT * FROM imagens WHERE fk_catalogo_id=c.cat_id LIMIT 3) i ON c.cat_id=i.fk_catalogo_id

 

Porém a coluna cat_id na subquery  (WHERE fk_catalogo_id=c.cat_idnão é reconhecida

Sabem se existe alguma forma de mudar isso? Tipo tornar a coluna Global.

 

Então tentei

SELECT (@valor_id:=c.`cat_id`) as id, c.`cat_nome_fantasia` as empresa, i.img_nome as imagem FROM `catalogo` c 
LEFT JOIN (SELECT * FROM imagens WHERE fk_catalogo_id=@valor_id LIMIT 3) i ON c.cat_id=i.fk_catalogo_id

 

 

Porém o @valor_id parece não ter ficado setado, pois não retornou registros.

 

Por fim tentei setar a variável.

SET @valor_id:=1;

SELECT (@valor_id:=c.`cat_id`) as id, c.`cat_nome_fantasia` as empresa, i.img_nome as imagem FROM `catalogo` c 
LEFT JOIN (SELECT * FROM imagens WHERE fk_catalogo_id=@valor_id LIMIT 3) i ON c.cat_id=i.fk_catalogo_id

 

Nesse caso quase funcionou, buscou as imagens com LIMIT.

Mas, ficou preso a setar manualmente o @valor_id, ou seja, só busca as imagens do id setado, o que não teria serventia pois seria o mesmo que inserir manualmente o id dentro do SELECT contido no JOIN.

 

Alguém tem alguma ideia de como resolver?

Obrigado

Compartilhar este post


Link para o post
Compartilhar em outros sites

Bem pessoal consegui uma solução simples, que não é exatamente o que procurava,
mas dá para quebrar um galho.
Vou continuar tentando encontrar a solução para a forma original que queria, usando (LIMIT) dentro do JOIN.

 

Objetivo:
Uso de 2 tabelas relacionadas 1:n onde temos vários imóveis na primeira tabela e cada imóvel possui várias imagens em outra tabela. 
Objetivo: Retornar apenas 5 imagens para cada imóvel em um SELECT com JOIN.

 

Solução provisória:
Ter na tabela imagens, além dos campos como (id, fk_id_imovel, nome_img, diretorio),
um campo ordem_imgs que deverá ser populado com números sequências iniciando no 1.

Assim no SELECT ficaria algo do tipo:

 

SELECT i.*, img.* FROM imoveis i 

LEFT JOIN imagens img ON img.fk_id_imovel=i.id AND img.ordem_imgs <=5 

 

Só para base de conhecimento:
Pode-se usar uma biblioteca JQuery  como (sortable) junto com o PHP e MySql para se ordenar as imagens arrastando-as (DRAG and Drop) no lado do cliente de modo a atualizar a sequência de números que determina a ordem das imagens na tabela do banco de dados.

Compartilhar este post


Link para o post
Compartilhar em outros sites

Essa situação poderia ser escrita sem JOIN mas apenas com a cláusula WHERE certo.

 

LIMIT não pode ser usado com JOIN.

Compartilhar este post


Link para o post
Compartilhar em outros sites
3 horas atrás, Bergs disse:

Essa situação poderia ser escrita sem JOIN mas apenas com a cláusula WHERE certo.

 

LIMIT não pode ser usado com JOIN.

 

Olá Bergs, realmente poderia usar o WHERE no lugar de  AND img.ordem_imgs <=5  

Ficaria: 

SELECT i.*, img.* FROM imoveis i 

LEFT JOIN imagens img ON img.fk_id_imovel=i.id 
WHERE img.ordem_imgs <=5 

 

O JOIN só permaneceria obviamente para o relacionamento das tabelas.

No caso eu dei preferencia para aproveitar o ON ao invés de usar o WHERE 

 

Quanto a LIMIT não poder ser usado com JOIN
como você disse,
eu estou cada vez mais ciente
de que possa ser impossível mesmo.

 

Acredito que por não ter uma maneira obvia de conseguir usar o LIMIT com JOIN
deve ser por não haver necessidade para isso,
afinal os DBs são bem elaborados e quando usamos chaves estrangeiras, índices,... 
imagino que o mysql já deva saber  que
o imóvel id=35 possua 12 imagens na tabela
ou que possua 5 imagens com com o campo (ordem_imgs) <=5.
Isso por causa do uso do índice, senão ele teria que 
verificar todos registros um a um para calcular isso.
Dessa forma, por já saber, é provável que ele já use o LIMIT automaticamente.

 

Suponho que é como se o mysql dissesse:  

 

"no meu índice indica 12 imagens para o id 35, encontrei elas, então encerro a busca por mais imagens com esse id e prossigo procurando as imagens relacionadas aos outros ids."

 

"no meu índice indica 5 imagens para o id 35 com o campo (ordem_imgs) tendo valor <=5, encontrei elas,
então encerro a busca por mais imagens com esse id e prossigo procurando as imagens relacionadas aos outros ids."

 

Não teria o porque após encontrar as 5 ou 12 imagens desejadas,
prosseguir verificando se encontra mais imagens
nas outras centenas ou milhares de linhas restantes na tabela.

 

Outra evidência é que ele tenha esse entendimento de quantas imagens existam relacionadas com determinado id,
é que quando usamos ON DELETE RESTRICT ele instantaneamente não permite deletar o registros que tenha outros relacionados. Será que ele varre toda a tabela para saber se tem tais registros (o que seria mais lento) ou usa os índices? Não tenho dúvidas que ele usará os índices.
 

Ou seja, o mysql é poderoso e inteligente 
e quando usamos chaves estrangeiras e índices,
ele organizará e usará os registros da melhor forma possível. 

 

Compartilhar este post


Link para o post
Compartilhar em outros sites
1 hora atrás, Didyo disse:

 

Olá Bergs, realmente poderia usar o WHERE no lugar de  AND img.ordem_imgs <=5  

Ficaria: 

SELECT i.*, img.* FROM imoveis i 

LEFT JOIN imagens img ON img.fk_id_imovel=i.id 
WHERE img.ordem_imgs <=5 

 

O JOIN só permaneceria obviamente para o relacionamento das tabelas.

No caso eu dei preferencia para aproveitar o ON ao invés de usar o WHERE 

 

Quanto a LIMIT não poder ser usado com JOIN
como você disse,
eu estou cada vez mais ciente
de que possa ser impossível mesmo.

 

Acredito que por não ter uma maneira obvia de conseguir usar o LIMIT com JOIN
deve ser por não haver necessidade para isso,
afinal os DBs são bem elaborados e quando usamos chaves estrangeiras, índices,... 
imagino que o mysql já deva saber  que
o imóvel id=35 possua 12 imagens na tabela
ou que possua 5 imagens com com o campo (ordem_imgs) <=5.
Isso por causa do uso do índice, senão ele teria que 
verificar todos registros um a um para calcular isso.
Dessa forma, por já saber, é provável que ele já use o LIMIT automaticamente.

 

Suponho que é como se o mysql dissesse:  

 

"no meu índice indica 12 imagens para o id 35, encontrei elas, então encerro a busca por mais imagens com esse id e prossigo procurando as imagens relacionadas aos outros ids."

 

"no meu índice indica 5 imagens para o id 35 com o campo (ordem_imgs) tendo valor <=5, encontrei elas,
então encerro a busca por mais imagens com esse id e prossigo procurando as imagens relacionadas aos outros ids."

 

Não teria o porque após encontrar as 5 ou 12 imagens desejadas,
prosseguir verificando se encontra mais imagens
nas outras centenas ou milhares de linhas restantes na tabela.

 

Outra evidência é que ele tenha esse entendimento de quantas imagens existam relacionadas com determinado id,
é que quando usamos ON DELETE RESTRICT ele instantaneamente não permite deletar o registros que tenha outros relacionados. Será que ele varre toda a tabela para saber se tem tais registros (o que seria mais lento) ou usa os índices? Não tenho dúvidas que ele usará os índices.
 

Ou seja, o mysql é poderoso e inteligente 
e quando usamos chaves estrangeiras e índices,
ele organizará e usará os registros da melhor forma possível. 

 

 

Isso mesmo Join para relacionamentos, acabei confundindo. :smiley:

 

Você tem essas tabelas  e como printar elas para entender melhor o uso do operador relacional no campo ordem_imgs <=5  porque não entendi a questão do filtro pela ordem. Seria um campo do tipo Primary Key com AUTO_INCREMENT   e por ele você fez o limite?

 

 

 

 

 

 

 

 

 

Compartilhar este post


Link para o post
Compartilhar em outros sites
6 horas atrás, Bergs disse:

Isso mesmo Join para relacionamentos, acabei confundindo. :smiley:

Você tem essas tabelas  e como printar elas para entender melhor o uso do operador relacional no campo ordem_imgs <=5  porque não entendi a questão do filtro pela ordem. Seria um campo do tipo Primary Key com AUTO_INCREMENT   e por ele você fez o limite?

 

Olá Bergs, fiz uma tabela simples apenas para demonstração. Segue em anexo uma imagem.

Quanto ao campo ordem_imgs não é do tipo Primary Key e nem AUTO_INCREMENT.

Tanto é que todos imóveis terão as imagens com a mesma numeração em sua ordem (1,2,3,4,...) havendo números repetidos, mas não para um mesmo imóvel.

 

Tem uma explicação junto a imagem.

Algo que poderia usar seria um Índice UNIQUE com os campos fk_imoveis_imv_id e img_ordem. Assim o DB não permitirá que determinado id de imóvel tenha números repetidos no campo img_ordem. 

 

Abraço e dúvidas ou sugestões não deixe de compartilhar.

 

InnerJoinComLimitImagens.jpg

Compartilhar este post


Link para o post
Compartilhar em outros sites

  • Conteúdo Similar

    • Por douglas79
      Bom dia,

      Há alguns dias que venho instalar o apache, o php, mysql e o phpmyadmin manualmente e sem obter sucesso. Até consegui rodar o php, porém, quando vou baixar a úitima versão do MYSQL, não tem todos os pacotes nele instalados, inclusive no completo, só encontro o Router.
      Alguém pode me dizer o porquê que isso está ocorrendo?
      Desde já agradeço a ajuda de vocês, que será bem vinda!
      No aguardo!

      Uso a versão 8.3.9 do PHP
      Meu SO é o Windows 10 32 bits
    • Por landerbadi
      Tenho uma tabela chamada "item" com os seguintes campos: id, name, active. Nela tem cadastrado vários itens. No campo "active" eu coloco a letra "S" para informar que este item está ativo no sistema. Por exemplo: 1, casa, S 2, mesa, S 3, cama, S 4, moto S 5, rádio O quinto registro "radio" não está ativo no sistema pois não tem um "S" no campo active. E outra tabela chamada "product" com os seguintes campos (id, name) com os seguintes registros: 1, Produto A 2, Produto B 3, Produto C E uma terceira tabela chamada "product_item" com os seguintes campos (productID, itemID). No campo productID eu coloco o id de um produto da tabela "product" e no campo "itemID" eu coloco o id do produto da tabela "item". Exemplo: 1, 1 1, 3 1, 4 2, 3 2, 4 Sendo assim o produto A da tabela 'product" comtem os itens casa, cama e moto. Eu preciso fazer uma busca da seguinte maneira:  Eu escolho um registro da tabela "item", por exemplo "casa". Preciso fazer com que o php me liste todos os registros da tabela "product" que contenham a palavra "casa" e que os demais itens estejam ativos no siste. Ou seja, que contenham um "S" no campo "active"  Eu consegui fazer isso da seguinte maneira: SELECT P.id, P.name, GROUP_CONCAT(I.name ORDER BY I.name) AS items FROM product P JOIN product_item PI ON P.id = PI.productID JOIN item I ON I.id = PI.itemID AND I.active = 'S' WHERE P.id NOT IN ( SELECT PI.productID FROM product_item PI JOIN item I ON I.id = PI.itemID WHERE I.active IS NULL ) AND P.id IN ( SELECT PI.productID FROM product_item PI JOIN item I ON I.id = PI.itemID WHERE I.name = 'mesa' ) GROUP BY P.id, P.name; O problema que eu estou tendo é o seguinte:
      Quando eu jogo este código para o banco de dados onde eu já tenho os registros cadastrado o php fica lendo uma eternidade e não lista os produtos.
       
      Usando código no banco de dados que eu fiz para testes ele funciona perfeitamente pois nele tem poucos registros.
       
      No banco de dados principal a tabela "item" tem 11.196 registros. A tabela "product" tem 88.214 registros e a tabela "product_item" tem 518.378 registros. 
       
      Eu acredito que, devido o banco de dados ser muito grande, ele não consegue listar.
       
      Alguém sabe de algum meio de resolver isso?
       
       
    • Por landerbadi
      Boa tarde pessoal. Estou tentado fazer uma consulta no banco de dados porém estou tendo dificuldades. Tenho uma tabela chamada "itens" com os seguintes campos: id, item, ativo. Nela tem cadastrado vários itens. No campo ativo eu coloco a letra "S" para informar que este item está ativo no sistema. Por exemplo: 1, casa, S 2, mesa, S 3, cama, S 4, moto S 5, rádio O quinto registro "radio" não está ativo no sistema pois não tem um "S" no campo ativo. E outra tabela chamada "produtos" com os seguintes campos (id, item1, item2, item3) com os seguintes registros: 1, casa, mesa, moto 2, mesa, casa, cama 3, rádio, cama, mesa Eu preciso fazer uma busca na tabela produtos da seguinte maneira: Eu escolho um registro na tabela "itens", por exemplo "mesa". Preciso fazer com que o php me liste todos os registros da tabela "produtos" que contenham a palavra "mesa". Até aqui tudo bem eu consigo listar. Estou fazendo assim: <?php $item = "mesa" $sql = mysqli_query($conn, "SELECT * FROM produtos WHERE item1 LIKE '$item' OR item2 LIKE '$item' OR item3 LIKE '$item' LIMIT 10"); while($aux = mysqli_fetch_assoc($sql)) { $id = $aux["id"]; $item1 = $aux["item1"]; $item2 = $aux["item2"]; $item3 = $aux["item3"]; echo $id . " - " . $item1 . ", " . $item2 . ", " $item3 . "<br>"; } ?> O problema é que está listando todos os registros que contém o item mesa. Eu preciso que o php verifique os demais item e me liste somente os registro em que todos os registros estejam ativos no sistema. No exemplo acima ele não deveria listar o registro 3. pois nesse registro contém o item "radio" e este item não está ativo no sistema. Ou seja, o registro "radio" na tabela itens não possui um "S" na coluna "ativo". Alguém sabe como resolver isso?
    • Por First
      Olá a todos!
       
      Eu estou criando um sistema do zero mas estou encontnrando algumas dificuldades e não estou sabendo resolver, então vim recorrer ajuda de vocês.
      Aqui está todo o meu código: https://github.com/PauloJagata/aprendizado/
       
      Eu fiz um sistema de rotas mas só mostra o conteúdo da '/' não sei porque, quando eu tento acessar o register nada muda.
      E eu também quero que se não estiver liberado na rota mostra o erro de 404, mas quando eu tento acessar um link inválido, nada acontece.
      Alguém pode me ajudar com isso? E se tiver algumas sugestão para melhoria do código também estou aceitando.
       
       
      Desde já, obrigado.
    • Por landerbadi
      Olá pessoal, boa tarde
       
      Tenho uma tabela chamada "produtos" com os seguintes campos (id, produto) e outra tabela chamada "itens" com os seguintes campos (id, prod_01, prod_02, prod_03, prod_04).
       
      Na tabela produtos eu tenho cadastrado os seguintes produtos: laranja, maçã, uva, goiaba, arroz, feijão, macarrão, etc.
       
      Na tabela itens eu tenho cadastrado os itens da seguinte maneira:
       
      1, laranja, uva, arroz, feijão;
      2, maçã, macarrão, goiaba, uva;
      3, arroz, feijão, maçã, azeite
       
      Meu problema é o seguinte: 
      Eu escolho um produto da tabela "produtos", por exemplo "uva".  Preciso fazer uma consulta na tabela "itens" para ser listado todos os registros que contenham o produto "uva" e que todos os demais produtos estejam cadastrados na tabela "produtos".
       
      No exemplo acima seria listado apenas dois registros, pois o terceiro registro não contém o produto "uva". 
       
      Alguém pode me ajudar? Pois estou quebrando a cabeça a vários dias e não consigo achar uma solução.
×

Informação importante

Ao usar o fórum, você concorda com nossos Termos e condições.