Ir para conteúdo

POWERED BY:

Arquivado

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

Renan Cabral

Distinct no MySQL

Recommended Posts

Caros,

Preciso de uma grande ajuda e não estou conseguindo avançar.

 

Tenho um banco de dados MySQL com 1 tabela que contem mais de 8.000.000 campos. Basicamente ela é assim (exemplo):

ID ID_Equipamento Valor1 Valor2
5000 10 fdas 341341
4999 9 fdas 15145
4998 8 fasfa 4351
4997 7 fdafd 415
4996 3 das 67467
4995 2 dasdasdas 7
4994 1 qgrgbsg 74
4993 9 bsgb 423527
4992 8 sgarhy 6
4991 6 bsbsgb 8461546
4990 5 rwrqrqrgaf 5
4989 4 fvafdv 87
4988 3 afvafg 85
4987 1 afvafg 85
4986 5 ag 634
4985 3 gfag 2
4984 2 fdaga 51
4983 9 gfaga 431
4982 8 gffga 245262653
4981 6 gfa 2

 

Onde ID é um auto incrementador na hora da inserção de um dado.

E ID_Equipamento é a referência de qual equipamento as informações são.

 

Estou precisando pegar os últimos valores (Valor1, Valor2) de cada ID_Equipamento (Todos os ID_Equipamentos - 1 a 10, mas somente o último cadastrado).

O último valor seria pelo último ID cadastrado daquele ID_Equipamento

Nesse caso seria:

ID ID_Equipamento Valor1 Valor2
5000 10 fdas 341341
4999 9 fdas 15145
4998 8 fasfa 4351
4997 7 fdafd 415
4996 3 das 67467
4995 2 dasdasdas 7
4994 1 qgrgbsg 74
       
4991 6 bsbsgb 8461546
4990 5 rwrqrqrgaf 5
4989 4 fvafdv

87

 

 

Já tentei utilizar o DISTINCT, porém ele só retorna uma coluna com o ID_Equipamento:

ID_Equipamento
10
9
8
7
3
2
1
6
5
4

 

 

Há alguma possibilidade de fazer o que eu quero fazer? Ou usando group by? ou MAX(ID)?

Depois que conseguir fazer no MySQL, irei passar pro PHP e assim o tempo não pode ser tão alto para a execução...

 

Grato!

Compartilhar este post


Link para o post
Compartilhar em outros sites

Sim é possível, seguem alguns meios de se fazer isso, sendo o primeiro preferível aos outros:

SELECT
  tabela.id,
  tabela.id_equipamento,
  tabela.valor1,
  tabela.valor2 
FROM
  tabela
INNER JOIN
  (SELECT
     MAX(id) AS id
   FROM
     tabela
   GROUP BY
     id_equipamento) AS ultimo_registro_por_equipamento
ON
  ultimo_registro_por_equipamento.id = tabela.id
ORDER BY
  tabela.id DESC
  
/*Também é possível alcançar o mesmo resultado com as seguintes consultas porem a menos que o banco consiga otimiza-las elas tendem a ter um desempenho inferior.*/

SELECT
  tabela.id,
  tabela.id_equipamento,
  tabela.valor1,
  tabela.valor2 
FROM
  tabela
WHERE
  id IN (SELECT
     MAX(id)
   FROM
     tabela
   GROUP BY
     id_equipamento)
ORDER BY
  tabela.id DESC
  
/*ou*/

SELECT
  t1.id,
  t1.id_equipamento,
  t1.valor1,
  t1.valor2 
FROM
  tabela AS t1
WHERE
  id = (SELECT
     MAX(t2.id)
   FROM
     tabela AS t2
   WHERE
     t2.id_equipamento = t1.id_equipamento)
ORDER BY
  t1.id DESC

O que elas fazem é consultar os maiores IDs da tabela por equipamento e depois buscar os dados com base neles, as consultas alternativas tendem a ser mais lentas pois a subquery será executada para cada registro no banco enquanto com o join ela será executada somente uma vez.

 

Você pode realizar testes em http://sqlfiddle.com/#!9/01f85c/11.

Compartilhar este post


Link para o post
Compartilhar em outros sites

Obrigado pela ajuda.

Mas suponho que pela quantidade de IDs (10 milhões) e pela quantidade de ID_Equipamento (300), ainda continua lento o sistema.

A primeira ideia nem consegui rodar, ficou carregando...

 

Obrigado mais uma vez...

Compartilhar este post


Link para o post
Compartilhar em outros sites

Se o Id e id_equipamento possuem índices então o problema está sendo cruzar os dados com o sub select, nesse caso você pode substitui-lo por uma view materializada, que nada mais é que uma tabela auxiliar que você ira alimentar antes da consulta e/ou via triggers sempre que houver alterações na tabela.

 

VIEW MATERIALIZADA

CREATE TABLE ultimos_lancamentos (
    id INTEGER PRIMARY KEY,
    id_equipamento INTEGER NOT NULL UNIQUE,
    FOREIGN KEY fk_ultimos_lancamentos_id (id) REFERENCES tabela(id) ON DELETE CASCADE,
    FOREIGN KEY fk_ultimos_lancamentos_id_equipamento (id_equipamento) REFERENCES equipamentos(id) ON DELETE CASCADE
);

E caso raramente vá consultá-la pode populá-la antes de cada consulta.

TRUNCATE ultimos_lancamentos;

INSERT INTO ultimos_lancamentos(id, id_equipamento)
    SELECT MAX(id), id_equipamento FROM tabela GROUP BY id_equipamento;

Se for consultá-la constantemente pode atualizá-la imediatamente criando um triggers ao inserir e atualizar(caso o id do equipamento possa ser alterado posteriormente) a chave estrangeira já cuida da remoção quando houver.

CREATE TRIGGER atualizar_ultimos_lancamentos_insert AFTER INSERT ON tabela
       FOR EACH ROW REPLACE ultimos_lancamentos(id, id_equipamento) VALUES(NEW.id, NEW.id_equipamento);

CREATE TRIGGER atualizar_ultimos_lancamentos_update AFTER UPDATE ON tabela
       FOR EACH ROW REPLACE ultimos_lancamentos(id, id_equipamento) VALUES(NEW.id, NEW.id_equipamento);

Independente do modo escolhido sua consulta ficaria assim:

SELECT
  tabela.id,
  tabela.id_equipamento,
  tabela.valor1,
  tabela.valor2 
FROM
  tabela
INNER JOIN
  ultimos_lancamentos
ON
  ultimos_lancamentos.id = tabela.id
ORDER BY
  tabela.id DESC

 

Compartilhar este post


Link para o post
Compartilhar em outros sites

  • Conteúdo Similar

    • Por violin101
      Caros amigos, saudações.
       
      Humildemente peço desculpa por postar uma dúvida que tenho.

      Preciso salvar no MySql, os seguinte Registro:

      1 - Principal
      ====> minha dúvida começa aqui
      ==========> como faço para o Sistema Contar Automaticamente o que estiver despois do 1.____?
      1.01 - Matriz
      1.01.0001 - Estoque
      1.01.0002 - Oficina
      etc

      2 - Secundário
      2.01 - Loja_1
      2.01.0001 - Caixa
      2.01.0002 - Recepção
      etc
       
      Resumindo seria como se fosse um Cadastro de PLANO de CONTAS CONTÁBEIL.

      Grato,


      Cesar









       
    • Por violin101
      Caros amigos, saudações.

      Por favor, me perdoa em recorrer a orientação dos amigos.

      Preciso fazer um Relatório onde o usuário pode Gerar uma Lista com prazo para vencimento de: 15 / 20/ 30 dias da data atual.

      Tem como montar uma SQL para o sistema fazer uma busca no MySql por período ou dias próximo ao vencimento ?

      Tentei fazer assim, mas o SQL me traz tudo:
      $query = "SELECT faturamento.*, DATE_ADD(faturamento.dataVencimento, INTERVAL 30 DAY), fornecedor.* FROM faturamento INNER JOIN fornecedor ON fornecedor.idfornecedor = faturamento.id_fornecedor WHERE faturamento.statusFatur = 1 ORDER BY faturamento.idFaturamento $ordenar ";  
      Grato,
       
      Cesar
       
       
       
       
    • Por violin101
      Caros amigos, saudações
       
      Por favor, me perdoa em recorrer a orientação dos amigos, tenho uma dúvida.
       
      Gostaria de uma rotina onde o Sistema possa acusar para o usuário antes dos 30 dias, grifar na Tabela o aviso de vencimento próximo, por exemplo:
       
      Data Atual: 15/11/2024
                                           Vencimento
      Fornecedor.....................Data.....................Valor
      Fornecedor_1...........01/12/2024..........R$ 120,00 <== grifar a linha de Laranja
      Fornecedor_1...........01/01/2025..........R$ 130,00
      Fornecedor_2...........15/12/2024..........R$ 200,00 <== grifar a linha de Amarelo
      Fornecedor_2...........15/01/2025..........R$ 230,00
      Fornecedor_3...........20/12/2024..........R$ 150,00
       
      Alguém tem alguma dica ou leitura sobre este assunto ?

      Grato,
       
      Cesar
    • Por violin101
      Caros amigos, saudações.

      Por favor, me perdoa em recorrer a ajuda dos amigos, mas preciso entender uma processo que não estou conseguindo sucesso.

      Como mencionado no Título estou escrevendo um Sistema Web para Gerenciamento de Empresa.
       
      Minha dúvida, que preciso muito entender:
      - preciso agora escrever a Rotina para Emissão de NFe e essa parte não estou conseguindo.
       
      tenho assistido alguns vídeos e leituras, mas não estou conseguindo sucesso, já fiz toda as importações das LIB da NFePhp conforme orientação.

      Preciso de ajuda.

      Algum dos amigos tem conhecimento de algum passo-a-passo explicando a criação dessa rotina ?

      tenho visto alguns vídeos com LARAVEL, mas quando tento utilizar e converter para PHP+Codeiginter, dá uma fila de erros que não entendo, mesmo informando as lib necessárias.

      Alguns do amigo tem algum vídeo, leitura explicando essa parte ?

      Grato,

      Cesar.
    • Por violin101
      Caros amigos, saudações.

      Por favor, me perdoa em recorrer ao auxílio dos amigos, mas preciso entender e resolver um problema.
       
      Tenho uma Rotina que o usuário seleciona os produtos que deseja para requerer ao setor responsável.
       
      O usuário escolhe um produto qualquer e Clicla em um button para incluir a lista.

      O problema que estou enfrentando é que após escolher o produto e teclar ENTER o Sistema já salva no BD.
       
      Gostaria de criar uma Tecla de Atalho, para quando incluir/escolher o produto na lista, o usuário tecla como exemplo:
      ALT+A  para agregar a lista
      ALT+S para salvar a lista de itens desejados.

      Assim, quando teclar enter, o sistema não dispara o GRAVAR na Base de Dados.

      Grato,

      Cesar
       
       
       
×

Informação importante

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