Foros Club Delphi

Foros Club Delphi (https://www.clubdelphi.com/foros/index.php)
-   SQL (https://www.clubdelphi.com/foros/forumdisplay.php?f=6)
-   -   Resultado de consulta evitando duplicados de ciertos registros (https://www.clubdelphi.com/foros/showthread.php?t=86009)

Sergio J. 05-06-2014 19:41:59

Resultado de consulta evitando duplicados de ciertos registros
 
Hola a todos, después de buscar un rato (quizás no he sabido buscar bien) no he encontrado ningún hilo que me saque de la duda.
Resulta que en una tabla de registros hay un campo que almacena una referencia y esta se puede repetir en varios registros.
Necesito una consulta que me devuelva sólo un registro (con todos sus campos) de cada valor diferente del campo referencia; intuyo que se utiliza el atributo distinct pero he hecho pruebas y no acierto.

También me vale una solución que teniendo en un ClientDataSet todos los datos, pueda realizar un filtrado y me quede con los registros que cumplan la condición mencionada antes, que muestre un registro por cada valor del campo referencia.

Gracias de antemano por vuestro tiempo y ayuda.

Sergio

Casimiro Notevi 05-06-2014 20:31:09

Cita:

Empezado por Sergio J. (Mensaje 477385)
intuyo que se utiliza el atributo distinct pero he hecho pruebas y no acierto.

Pues pon el código que lo veamos.
Y explica qué quiere decir con "no acierto", ¿en qué no aciertas?

Sergio J. 06-06-2014 07:54:08

Bien, en la tabla lo que tengo es esto:

Código:

id.    referencia  campo1  campo2...
1        0101          23      12
2        0101          54      32
3        0101          43      43
4        0102          12      54
5        0103          32      45
...


Dados estos datos, me interesa una consulta tal que al aplicarla, me quede con los registros en los que no se repita el campo referencia, como en este caso:


Código:

id.    referencia  campo1  campo2...
1        0101          23      12
4        0102          12      54
5        0103          32      45
...


He probado algo así como:

Código SQL [-]
select (dintinct referencia), t.*
from tabla t
where condiciones

pero no ha funcionado.

O también me interesa que a la hora de filtrar los datos en la aplicación, en el evento OnFilterRecord, haga lo mismo pero no encuentro la forma.

A ver si alguien me echa una mano, gracias.

Sergio

engranaje 06-06-2014 08:30:19

Puedes provar algo del tipo:

select * from tabla t1 where ((select count(*) from tabla t2 where t1.referenca=t2.referencia)<2 )

newtron 06-06-2014 09:49:00

Hola.

Yo no soy experto en sql pero prueba a añadir a tu instrucción la cláusula "GROUP BY REFERENCIA"

Saludos

duilioisola 06-06-2014 09:59:03

Creo que esta es la consulta que quieres:

Código SQL [-]
/*
Traigo todos los registros de tabla
donde el id corresponda a la primera ocurrencia de cada referencia
*/
select *
from tabla
where
id in (select min(id)
       from tabla
       group by referencia)

Casimiro Notevi 06-06-2014 10:01:56

Cita:

Empezado por engranaje (Mensaje 477412)
Puedes provar algo del tipo:
select * from tabla t1 where ((select count(*) from tabla t2 where t1.referenca=t2.referencia)<2 )

Recuerda poner los tags al código fuente, ejemplo:



Gracias :)


Cita:

Empezado por newtron (Mensaje 477413)
Hola.
Yo no soy experto en sql pero prueba a añadir a tu instrucción la cláusula "GROUP BY REFERENCIA"
Saludos

Exacto.

El "problema" es que aunque se ponga distinct, al añadir el asterisco para el resto de campos, finalmente se trae todos, por eso hay que agrupar por ese campo "referencia".
En este caso no hace falta hacer un distinct, solamente es necesario agrupar:
Código SQL [-]
select * from tabla group by referencia

Sergio J. 06-06-2014 10:20:22

Cita:

Empezado por engranaje (Mensaje 477412)
Puedes provar algo del tipo:

select * from tabla t1 where ((select count(*) from tabla t2 where t1.referenca=t2.referencia)<2 )

Gracias engranaje pero de esta manera no devuelve los registros que tienen más de un registro con la misma referencia.
Me devolvería esto:

Código:

id.    referencia  campo1  campo2...
4        0102          12      54
5        0103          32      45
...

...y creo que newtron tampoco ha acertado ya que al introducir un group by, en la sentencia select me obliga a poner todos los campos de la tabla y aún así tampoco resultaría...

Código:

select distinct a.referencia, count(a.referencia)
from tabla a
group by referencia

...esto funciona pero quiero que muestre todos los demás campos de la tabla.
Sin embargo si añado a la sentencia select todos los demás campos no funciona:

Código:

select distinct a.referencia, count(a.referencia), a.*
from tabla a
group by referencia

A esto me refería con lo de que "no acierto".

Sergio J. 06-06-2014 10:25:42

Cita:

Empezado por duilioisola (Mensaje 477415)
Creo que esta es la consulta que quieres:

Código SQL [-]
/*
Traigo todos los registros de tabla
donde el id corresponda a la primera ocurrencia de cada referencia
*/
select *
from tabla
where
id in (select min(id)
       from tabla
       group by referencia)

Efectivamente, esta sería la consulta efectiva, ya que el resultado es el esperado pero se ralentiza un poco.
Por otro lado quería aclarar a casimiro que la consulta

Código SQL [-]
select * from tabla group by referencia

pide a gritos un campo por el cual agrupar y daría error.

Si no encuentro otra consulta sin anidaciones, cosa que creo sería imposible, pues me quedo con la de duilioisola que es la que me soluciona el problema.

Gracias a todos por vuestro tiempo y aportaciones.

Sergio J.

aposi 06-06-2014 10:40:10

Cita:

Empezado por engranaje (Mensaje 477412)
Puedes provar algo del tipo:

select * from tabla t1 where ((select count(*) from tabla t2 where t1.referenca=t2.referencia)<2 )

Con esta solo devolveria los que no estan repetidos

prueba esto

Código SQL [-]
select * from tabla t1 where id = (select min(id) from tabla t2  where t1.referenca=t2.referencia )

Con esta consulta tendria que retornar lo que quieres

engranaje 06-06-2014 11:46:52

Primero disculparme por no haber leido bien la pregunta y no darme cuenta de que necesitabas también una ocurrencia para cada registro repetido. Segundo seguir discupandome, esta vez por la omisión del tag sql y por haber escrito prueva en lugar de prueba, cuando quise editar para corregir ya no era posible.

Ahora la aportación que igual te sirve. Puede que con esta consulta obtengas lo que necesites con la velocidad que te gustaría:
Código SQL [-]
  SELECT t1.*
  FROM tabla AS t1
  LEFT OUTER JOIN tabla AS t2
 ON t1.referencia = t2.referencia  
 AND t1.id < t2.id
 WHERE t2.referencia IS NULL

Sergio J. 09-06-2014 12:21:08

Gracias de nuevo a todos, con todas estas últimas consultas, ya sólo me queda elegir la más eficiente porque funcionan todas.

Un saludo

Sergio

Sergio J. 11-06-2014 08:09:33

Tema solucionado

newtron 11-06-2014 09:29:01

Cita:

Empezado por Sergio J. (Mensaje 477638)
Tema solucionado

¿Y nos vas a dejar con la intriga de qué has hecho al final? :D

Sergio J. 11-06-2014 09:39:31

Bueno, la consulta que ha prevalecido a las demás es la siguiente de aposi:

Código SQL [-]
select * from tabla t1 where id = (select min(id) from tabla t2  where t1.referenca=t2.referencia )

...a pesar de ser anidada, se ejecuta rápido y el resultado es exactamente el esperado.

Gracias a todos, de verdad.

Sergio J.

duilioisola 11-06-2014 10:17:06

Una pequeña nota sobre la consulta:

Dependiendo de los índices (ascendente, descendente) probablemente la consulta mejore si pones max(id). Obviamente el registro será del último id y no del primero.
Si no te importe cuál de los registros salga, puedes probar con FIRST 1 id. Esto te seleccionará uno de los registros, pero no te asegura cual de todos. Se lo supone más óptimo porque en cuanto encuentra 1 deja de buscar. En los otros casos debe buscar en todos los que se cumplan la condición (t1.referencia=t2.referencia) y luego seleccionar uno.

Código SQL [-]
/* el primer registro de los que se repiten (ordenado por id) */
select * from tabla t1 where id = (select min(id) from tabla t2  where t1.referenca=t2.referencia)

/* el último registro de los que se repiten (ordenado por id) */
select * from tabla t1 where id = (select max(id) from tabla t2  where t1.referenca=t2.referencia)

/* uno de los registros de los que se repiten */
select * from tabla t1 where id = (select first 1 id from tabla t2  where t1.referenca=t2.referencia)

Sergio J. 11-06-2014 12:20:51

Cita:

Empezado por duilioisola (Mensaje 477646)
Una pequeña nota sobre la consulta:

...
Código SQL [-]
...

/* uno de los registros de los que se repiten */
select * from tabla t1 where id = (select first 1 id from tabla t2  where t1.referenca=t2.referencia)
...

Efectivamente, va más rápido y no me importa cual de los registros con la misma referencia me devuelva. No sabía de la sentencia "first 1", no la había utilizado nunca.

Gracias


La franja horaria es GMT +2. Ahora son las 06:44:34.

Powered by vBulletin® Version 3.6.8
Copyright ©2000 - 2024, Jelsoft Enterprises Ltd.
Traducción al castellano por el equipo de moderadores del Club Delphi