Foros Club Delphi

Foros Club Delphi (https://www.clubdelphi.com/foros/index.php)
-   Firebird e Interbase (https://www.clubdelphi.com/foros/forumdisplay.php?f=19)
-   -   SELECT con SUM y COUNT (https://www.clubdelphi.com/foros/showthread.php?t=93179)

Angel.Matilla 07-06-2018 19:40:37

SELECT con SUM y COUNT
 
Estoy teniendo un problema al que no sé como meterle mano. Tego este query:
Código PHP:

SELECT B.NombreCOUNT(C.Codigo), SUM(D.Censo)
  
FROM DatLoc AJunta BPersona CMesas D
 WHERE A
.Codigo B.Codigo 
   
AND A.Codigo C.Junta 
   
AND C.Situacion IN (12)
   AND 
A.Codigo D.Municipio
   
AND A.Codigo 39
 GROUP BY 1 

Vamos por partes. Digamos que "contando a pelo" sé que el resultado del COUNT para esos datos es de 907 y el SUM tiene que salir 58211. Eso es así sí o sí.


Si hago el SELECT sólo con el COUNT (SELECT B.Nombre, COUNT(C.Codigo)) o sólo con el SUM (SELECT B.Nombre, SUM(D.Censo)) me salen los resultados que he puesto más arriba, pero si combino ambas operaciones tal como en el código me da como resultado 90.700 para el COUNT y 52.797.377 para el SUM. Y me estoy volviendo loco tratando de averiguar porqué salen esos resultados.

manelb 07-06-2018 20:04:28

Yque resultado muestra si pones
Código SQL [-]
SELECT B.Nombre, COUNT(*), SUM(D.Censo)
  FROM DatLoc A, Junta B, Persona C, Mesas D
 WHERE A.Codigo = B.Codigo 
   AND A.Codigo = C.Junta 
   AND C.Situacion IN (1, 2)
   AND A.Codigo = D.Municipio
   AND A.Codigo = 39
 GROUP BY 1

Angel.Matilla 07-06-2018 20:05:29

Cita:

Empezado por manelb (Mensaje 526915)
Yque resultado muestra si pones

Lo mismo: 90700 y 52797377

Angel.Matilla 07-06-2018 20:07:47

El problema creo que ha de estar en como relacionar las tablas Junta y Mesas, que al poner ambos contadores pasa más de una vez por cada registro. Si no es así, no lo entiebdo.

mamcx 07-06-2018 20:12:13

Por lo tanto debes poner el codigo en el group by. (y pon un order by para que la consulta sea estable)

La otra, pon un ejemplo de los datos con valores duplicados pa que veamos como estan...

manelb 07-06-2018 20:19:18

Cita:

Empezado por mamcx (Mensaje 526919)
(y pon un order by para que la consulta sea estable)

Perdona mi ignorancia:) pero, ¿podrías comentar esto??

duilioisola 07-06-2018 20:58:11

El SQL que envías es teoricamente igual a este con JOINS. Pruébalo y nos cuentas...
Código SQL [-]
select b.nombre, count(codigo), sum(d.censo)
from datloc a
join junta b on a.codigo = b.codigo
join persona c on a.codigo = c.junta
join mesas d on a.codigo = d.municipio
where
c.situacion in (1, 2) and
a.codigo = 39
group by 1

mamcx 07-06-2018 22:26:01

Cita:

Empezado por manelb (Mensaje 526920)
Perdona mi ignorancia:) pero, ¿podrías comentar esto??




Sql NO GARANTIZA EL ORDEN DE LOS RESULTADOS.


Puedes obtener un resultado diferente con el tiempo, por ejemplo, si se mueven las filas o se cambian de puesto. Ademas los optimizadores pueden insertar ORDERS BY como lo vean que puede no concordar con lo que piensas.


A menos que tengas un CLUSTERED INDEX o un SORT BY definido no se puede aseverar que el resultado sera estable.

Angel.Matilla 08-06-2018 11:05:41

Gracias por viestras respuestas.

Cita:

Empezado por duilioisola (Mensaje 526921)
El SQL que envías es teoricamente igual a este con JOINS. Pruébalo y nos cuentas...

Da el mismo resultado absurdo.
Cita:

Empezado por mamcx (Mensaje 526919)
Por lo tanto debes poner el codigo en el group by. (y pon un order by para que la consulta sea estable)

Vamos a ver: un order by no tiene sentido porque la consulta ha de devolver una única fila. Y lo que necesito que me expliques es como meter el código en el group by teniendo en cuenta que código ESTÁ en un de los campos agrupados, en el COUNT.
Cita:

Empezado por mamcx (Mensaje 526919)
La otra, pon un ejemplo de los datos con valores duplicados pa que veamos como estan...

¿El resultado de la consulta? Claro:
Con el código que puse yo en el primer mensaje sale esto:
Cita:

Nombre COUNT SUM
CIUDAD REAL 90.700 52.797.377
Con el código que me propone duilioisola:
Cita:

Nombre COUNT SUM
CIUDAD REAL 90.700 52.797.377
Y tendría que salir esto:
Cita:

Nombre COUNT SUM
CIUDAD REAL 907 52.811

duilioisola 08-06-2018 12:15:59

Te sugiero hacer un SQL que te devuelva las lineas que va a sumar (Básicamente quitar el GROUP BY) y ver que sale.
Seguramente veras que se repite 10 veces algún "Identificador" por cada "NOMBRE".
Código SQL [-]
select 
/* Datos que se cuentan y suman */
b.nombre, a.codigo, d.censo
/* Datos que seguramente se repiten */
, b.IDENTIFICADOR_DE_JUNTA, c.IDENTIFICADOR_DE_PERSONA, d.IDENTIFICADOR_DE_MESAS
from datloc a
join junta b on a.codigo = b.codigo
join persona c on a.codigo = c.junta
join mesas d on a.codigo = d.municipio
where
c.situacion in (1, 2) and
a.codigo = 39
Por los resultados que muestras, me docy cuenta que está multiplicando por 10 el valor que consideras correcto.
Esto implica que cada "NOMBRE" se está repitiendo 10 veces con esta forma de unir las tablas.
Yo supongo que debe haber
  • 10 "JUNTA" en cada "DATLOC"
  • 10 "PERSONA" en cada "DATLOC"
  • 10 "MESAS" en cada "DATLOC"

Si este es el problema, deberás corregir el SQL para que te devuelva 1 solo registro por cada DATLOC.

Angel.Matilla 08-06-2018 12:51:53

Estoy corrigiendo ahora la estructira de las tabals (me he dado cuenta que me faltan algunas columnas en las definiciones), pero en cuanto termine de adaptarlas probaré lo que me dices. Gracias.

Angel.Matilla 11-06-2018 14:44:14

Cita:

Empezado por duilioisola (Mensaje 526940)
Te sugiero hacer un SQL que te devuelva las lineas que va a sumar (Básicamente quitar el GROUP BY) y ver que sale.
Seguramente veras que se repite 10 veces algún "Identificador" por cada "NOMBRE".

Para no variar tenías razón. Al introducir ese campo Codigo de la tabla Persona me devuelve una línea por cada una de las filas de esa tabla que cumple la condición AND C.Situacion IN (1, 2). Es evidente que ese campo no puede ir en el SELECT.


No obstante, como comentaba el otro día, he tenido que modificar un poco la estructura de las tablas (básicamente he tenido que añadir un nuevo campo en varias de ellas), pero sigo sin conseguir lo que busco. He cambiado el query a este otro, ya con las tablas modificadas, para comprobar la sugerencia de duilioisola:
Código PHP:

SELECT DISTINCT B.NombreD.DistritoD.SeccionD.MesaD.Censo
  FROM DatLoc A
Junta BPersona CMesas D
 WHERE A
.Municipio B.Municipio
   
AND A.Municipio C.Poblacion
   
AND C.Situacion IN (34)
   AND 
A.Municipio D.Municipio
   
AND A.Municipio 14921
 ORDER BY 2
,3,

Y ahora obtengo esto (un trozo):

En total salen 100 líneas, que es correcto y los datos que muestra están bien. Pero si ahora si hago SÓLO el SUM(D.Censo) con SELECT DISTINCT B.Nombre, SUM(D.Censo), en vez de los 58211 que debería dar salen los 52 millones. Perplejo estoy.

mamcx 11-06-2018 20:14:04

Pero nada que pones los datos iniciales, solo los resultados.

Haznos unas tablas con ejemplos representativos de las filas.

duilioisola 11-06-2018 20:27:48

Supongo entonces, que deberás hacer subselects:
Básicamente, haces un select que te devuelva lo que quieres (SELECT que solo devuelve nombre de junta agrupado).
Por cada registro haces un select que en el where filtre lo que corresponda con ese registro.
Es importante que solo devuelve un solo resultado. (SELECT SUM(), COUNT(), FIRST 1)

Ejemplo:
Código SQL [-]
select distinct b.nombre, d.distrito, d.seccion, d.mesa, d.censo,
                /* Personas en ese municipio */
                (select count(*)
                 from persona pp
                 where
                 pp.poblacion = a.municipio and
                 c.situacion in (3, 4)) cantidad_personas_municipio,
                /* Suma de xxx de las personas en ese municipio */
                (select sum(xxx)
                 from persona pp
                 where
                 pp.poblacion = a.municipio and
                 c.situacion in (3, 4)) suma_xxx,
from datloc a, junta b, persona c, mesas d
where
a.municipio = b.municipio and
a.municipio = c.poblacion and
c.situacion in (3, 4) and
a.municipio = d.municipio and
a.municipio = 14921
order by 2, 3, 4

Angel.Matilla 12-06-2018 10:45:05

Gracias por la idea. Estaba dándole vueltas a algo similar desde ayer pero no se me ocurría como. Voy a probarlo ahora mismo.

Angel.Matilla 12-06-2018 10:55:20

Cita:

Empezado por duilioisola (Mensaje 527022)
Supongo entonces, que deberás hacer subselects

¡Perfecto! Me ha quedado así:
Código PHP:

SELECT C.Nombre
       (
SELECT COUNT(*) FROM Persona E
         WHERE A
.Municipio E.Poblacion
           
AND E.Situacion IN (34)) cantidad_personas,
       (
SELECT SUM(F.CensoFROM Mesas F
         WHERE A
.Municipio F.Municipiocenso_electoral
  FROM DatLoc A
Junta BPoblacion C
 WHERE A
.Municipio B.Municipio
   
AND B.Municipio C.Codigo
   
AND A.Municipio 14921 

Muchísimas gracias. No sabes las vueltas que le he dado a este query.

duilioisola 12-06-2018 11:47:37

Sugerencia de estilo:

Código SQL [-]
/*
Utiliza un alias que ayude a saber qué tabla es la asociada. Es mas facil leer el sql.
En SQLs pequeños el alias podría ser incluso la primera letra de la tabla (d, j, p).
dat -> datloc
jun -> junta
pob -> poblacion
*/
select pob.nombre,
       (select count(*)
        from persona
        where
        poblacion = dat.municipio and
        situacion in (3, 4)) cantidad_personas,
       (select sum(censo)
        from mesas
        where
        municipio = dat.municipio) censo_electoral
from datloc dat, junta jun, poblacion pob
where
dat.municipio = jun.municipio and
jun.municipio = pob.codigo and
dat.municipio = 14921

Casimiro Notevi 12-06-2018 12:11:31

Cita:

Empezado por duilioisola (Mensaje 527036)
Sugerencia de estilo:

^\||/^\||/^\||/

Angel.Matilla 13-06-2018 11:19:36

Cita:

Empezado por duilioisola (Mensaje 527036)
Sugerencia de estilo:

No te quepa duda que la tendré en cuenta.


La franja horaria es GMT +2. Ahora son las 13:42:09.

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