* * * *
13 Visitatori, 0 Utenti

Autore Topic: [Risolto] SUM condizionale in espressione SQL  (Letto 122 volte)

Maxx

  • Newbie
  • *
  • Post: 30
  • Karma: +1/-0
[Risolto] SUM condizionale in espressione SQL
« il: Giugno 10, 2018, 08:01:36 pm »
Buonasera.
Quesito tecnico su istruzione SQL.
Ho una tabella fatta più o meno così:
Campo1 - Campo2 - ... - Data (Timestamp) - Valore (Single) - Tipo (Stringa 'A' o 'B' o altri valori da escludere) - ...

Volendo creare una tabella con le seguenti colonne:
-----------------------------------------------------------------------------------------
Data         | Somma valori per tipo A    | Somma valori per tipo B    | Totale A + B


Qual è l'istruzione sql più idonea? Io ci sono arrivato loopando per 7 volte (la tabella rappresenta il valore "incasso" della settimana)

   'Select SUM(Valore) FROM Incassi r ' +
     'WHERE CAST(r.Data_Mov as DATE) = :Dat_Calc ' +
     'AND r.Tipo_Pagam = :Pag_Tipo';

   ModDati.Qry_Pag.ParamByName('Dat_Calc').AsDate   := p_Ini;   
   ModDati.Qry_Pag.ParamByName('Pag_Tipo').AsString := 'A'; //Pagamento a mezzo Cash
   ModDati.Qry_Pag.Open;

   'Select SUM(Valore) FROM Incassi r ' +
     'WHERE CAST(r.Data_Mov as DATE) = :Dat_Calc ' +
     'AND r.Tipo_Pagam = :Pag_Tipo';

   ModDati.Qry_Pag.ParamByName('Dat_Calc').AsDate   := p_Ini;   
   ModDati.Qry_Pag.ParamByName('Pag_Tipo').AsString := 'B'; //Pagamento a mezzo Carte di credito
   ModDati.Qry_Pag.Open;

Ovviamente la mia idea era quella di fare un
o---------------------o
Select  CAST(r.Data_Mov as DATE),
   Sum(Valore) Per Tipo_Pagam = 'A' AS Val_Cash,
   Sum(Valore) Per Tipo_Pagam = 'B' AS Val_CdC,
   Val_Cash + Val_CdC AS Totale
From Incassi
Where Data_Mov > :Data_Ini and Data_Mov < :Data_Fin
Group by CAST(r.Data_Mov as DATE)
o---------------------o

Spero di essermi spiegato decentemente...

Grazie, Maxx
« Ultima modifica: Giugno 12, 2018, 06:33:18 pm da Maxx »

xinyiman

  • Administrator
  • Hero Member
  • *****
  • Post: 2415
  • Karma: +6/-0
Re:SUM condizionale in espressione SQL
« Risposta #1 il: Giugno 11, 2018, 10:34:21 am »
Ciao Maxx, ed esattamente cosa non ti funziona di quella query? Che errore ti ritorna?
Ieri è passato, domani è futuro, oggi è un dono...

Maxx

  • Newbie
  • *
  • Post: 30
  • Karma: +1/-0
Re:SUM condizionale in espressione SQL
« Risposta #2 il: Giugno 11, 2018, 10:41:11 am »
Ciao xinyi
Semplicemente non funziona! :)
Provato con where, con having, con in... Immagino sia questione di grammatica SQL di cui ho precaria conoscenza.
Nelle "prove" con SQL manager mi ritorna un errore appena dopo il Sum(Valore). Immagino che la struttura CASE sia diversa da quella che conoscevo io e non riesco a replicarla in FireBird.

Grazie per l'attenzione, xinyi.

xinyiman

  • Administrator
  • Hero Member
  • *****
  • Post: 2415
  • Karma: +6/-0
Re:SUM condizionale in espressione SQL
« Risposta #3 il: Giugno 11, 2018, 10:44:48 am »
Allegaci la query corretta, perchè

"Select  CAST(r.Data_Mov as DATE),
   Sum(Valore) Per Tipo_Pagam = 'A' AS Val_Cash,
   Sum(Valore) Per Tipo_Pagam = 'B' AS Val_CdC,
   Val_Cash + Val_CdC AS Totale
From Incassi
Where Data_Mov > :Data_Ini and Data_Mov < :Data_Fin
Group by CAST(r.Data_Mov as DATE)"

è troppo generica. Inoltre non è il caso di fare quei cast se il dato è già corretto.
Ieri è passato, domani è futuro, oggi è un dono...

Maxx

  • Newbie
  • *
  • Post: 30
  • Karma: +1/-0
Re:SUM condizionale in espressione SQL
« Risposta #4 il: Giugno 11, 2018, 10:55:29 am »
Il cast serve (almeno nel grouping), altrimenti essendo il campo un type Timestamp non raggruppa nulla, considerando per ogni millesimo di secondo un record di group diverso.

o----------o

Per quel che riguarda l'istruzione, dovrebbe essere proprio simile a questa (a parte il *Per, ovviamente):

"Select  CAST(r.Data_Mov as DATE),
   Sum(Valore) *Per Tipo_Pagam = 'A' AS Val_Cash,
   Sum(Valore) *Per Tipo_Pagam = 'B' AS Val_CdC,
   Val_Cash + Val_CdC AS Totale
From Incassi
Where Data_Mov > :Data_Ini and Data_Mov < :Data_Fin
Group by CAST(r.Data_Mov as DATE)

Da una tabella incassi, voglio sommare, raggruppati per data, tutti i valori del campo "Valore" divisi per tipologia di pagamento (mi servono altre distinzioni, ma una volta appreso, il procedimento sarebbe lo stesso).

xinyiman

  • Administrator
  • Hero Member
  • *****
  • Post: 2415
  • Karma: +6/-0
Re:SUM condizionale in espressione SQL
« Risposta #5 il: Giugno 11, 2018, 11:05:26 am »
Se ti riferisci al case di postgres allora su firebird puoi usare iif

https://firebirdsql.org/refdocs/langrefupd20-iif.html

Per il raggruppamento della data ti consiglio di usare il comando extract

https://firebirdsql.org/refdocs/langrefupd21-intfunc-extract.html

estrai l'anno, il mese e la data e poi tramite il cast li trasformi in stringhe e le accodi in modo da avere una stringa del tipo YYYYMMDD e fai il group by su quello. Però notavo che tu non raggruppi per tipo di pagamento e quindi il sum ti sommerà sempre tutto e poi lo moltiplica per il tipo di pagamento, così non potrà mai funzionare. Questo perchè vuoi mettere i valori sulle singole colonne.
Ieri è passato, domani è futuro, oggi è un dono...

Maxx

  • Newbie
  • *
  • Post: 30
  • Karma: +1/-0
Re:SUM condizionale in espressione SQL
« Risposta #6 il: Giugno 11, 2018, 11:21:26 am »
L' IIF fa una scelta su ogni singolo record. A me invece servono due somme, riferite al campo valore, di TUTTI i record ma divisi per tipo di pagamento (e, in secundis, per data). Non sono riuscito a trovare esempi che mi instradino correttamente, ma immagino che sia una cosa necessaria e che pertanto esista.
Il loop riesce a risolvere, ma di certo non è la soluzione più elegante, pratica e performante.

Per quel che riguarda il cast, il risultato della mia versione non dà alcun problema e rende pertanto pleonastico (e probabilmente più oneroso) l'uso di EXTRACT.

Stilgar

  • Global Moderator
  • Hero Member
  • *****
  • Post: 1772
  • Karma: +4/-0
Re:SUM condizionale in espressione SQL
« Risposta #7 il: Giugno 11, 2018, 01:49:47 pm »
Ciao.Potresti usare le sub query per ottenere il risultato che vuoi.
Codice: [Seleziona]
select data, a, b, a+b
from (
select max(data) data, sum(importo) a, 0 b
from incassi
where tipo ='A'
group by tipo
union
select max(data) data, 0 a, sum(importo) b
from incassi
where tipo ='B'
group by tipo
) a
order by data;
Il Case:
Codice: [Seleziona]
CASE <expression> WHEN <exp1> THEN result1 WHEN <exp2> THEN result2 ... [ELSE defaultresult] END
Codice: [Seleziona]
CanVote = case when Age >= 18 then 'Yes' when Age <  18 then 'No' else 'Unsure' end;
Al posto della tabella potresti usare una vista.In modo da permettere al db engine di ottimizzare in qualche modo l'accesso.:)
Stilgar
Al mondo ci sono 10 tipi di persone ... chi capisce il binario e chi no.

Maxx

  • Newbie
  • *
  • Post: 30
  • Karma: +1/-0
Re:SUM condizionale in espressione SQL
« Risposta #8 il: Giugno 11, 2018, 04:28:40 pm »
Grazie stilgar, disponibile come sempre.

In effetti avevo pensato ad una vista, ma ho il solito problema che non voglio "sporcare" la base dati proprietaria. E' possibile creare viste "temporanee" che non vadano a toccare l'fdb? Vorrei evitare di creare e poi distruggere entità "fisiche".

Grazie.

Stilgar

  • Global Moderator
  • Hero Member
  • *****
  • Post: 1772
  • Karma: +4/-0
Re:SUM condizionale in espressione SQL
« Risposta #9 il: Giugno 11, 2018, 04:30:00 pm »
Per la creazione di "oggetti temporanei" non ho idea se fbd possa supportarle in una qualche versione beta.Non mi risulta attualmente. :(

Stilgar
Al mondo ci sono 10 tipi di persone ... chi capisce il binario e chi no.

Maxx

  • Newbie
  • *
  • Post: 30
  • Karma: +1/-0
Re:SUM condizionale in espressione SQL
« Risposta #10 il: Giugno 11, 2018, 06:23:08 pm »
Grazie stilgar, grazie xinyiman, ci sono riuscito! :)

SELECT  CAST(d.Data_Mov as DATE),
  SUM(CASE when d.Tipo_Pagam = :Ty_Plas THEN d.Valore ELSE 0 END) as Plastic_sum,
  SUM(CASE when d.Tipo_Pagam = :Ty_Cash THEN d.Valore ELSE 0 END) as Cash_sum,
  SUM(d.AMOUNT) as Total_sum
  FROM Incassi d
  WHERE CAST(d.Data_Mov as DATE) >= :Data_Ini
      AND CAST(d.Data_Mov as DATE) <= :Data_Fin
  GROUP by CAST(d.Data_Mov as DATE)

Magari può essere utile a qualcun altro!

Grazie ancora,
  Maxx.

 

Recenti

How To

Trascinare un file nel programma da xinyiman
DB concetti fondamentali e ZeosLib da xinyiman
Recuperare codice HTML da pagina web da xinyiman
Mandare mail con Lazarus da xinyiman
Stabilire il sistema operativo da xinyiman
Esempio lista in pascal da xinyiman
File INI da xinyiman
Codice di attivazione da xinyiman
Realizzare programmi multilingua da xinyiman
Lavorare con le directory da xinyiman
Utilizzare Unità esterne da Loryea
TTreeView da xinyiman
TTreeview e Menu da xinyiman
Generare controlli RUN-TIME da xinyiman
LazReport, PDF ed immagini da xinyiman
Intercettare tasti premuti da xinyiman
Ampliare Lazarus da xinyiman
Lazarus e la crittografia da xinyiman
System Tray con Lazarus da xinyiman
UIB: Unified Interbase da Microges2000
Il file: questo sconosciuto da Microges2000
Conferma di chiusura di un applicazione da xinyiman
Liste e puntatori da Microges2000
Overload di funzioni da Microges2000
Funzioni a parametri variabili da Microges2000
Proprietà da Microges2000
Conversione numerica da Microges2000
TImage su Form e Panel da Maverich
Indy gestiore server FTP lato Client da Maverich
PopUpMenu sotto Pulsante (TSpeedButton) da Maverich
Direttiva $macro da Microges2000
Toolbar da xinyiman
Evidenziare voci TreeView da Maverich
Visualizzare un file Html esterno da Maverich
StatusBar - aggirare l'errore variabile duplicata da Maverich
Da DataSource a Excel da xinyiman
Le permutazioni da xinyiman
Brute force da xinyiman
Indy 10 - Invio email con allegati da Maverich
La gestione degli errori in Lazarus da xinyiman
Pascal Script da xinyiman
Linux + Zeos + Firebird da xinyiman
Dataset virtuale da xinyiman
Overload di operatori da Microges2000
Lavorare con file in formato JSON con Lazarus da nomorelogic
Zeos ... dietro le quinte (prima parte) da Stilgar
Disporre le finestre in un blocco unico (come Delphi) da Maverich
Aspetto retrò (Cmd Line) da xinyiman
Come interfacciare periferica twain da Narciso
Ubuntu - aggiornare free pascal e lazarus da xinyiman
fpcup: installazioni parallele di lazarus e fpc da nomorelogic
Free Pascal e Lazarus sul Raspberry Pi da nomorelogic
Cifratura: breve guida all'uso dell'algoritmo BlowFish con lazarus e free pascal. da nomorelogic
Creare un server multithread da xinyiman
guida all'installazione di fpc trunk da subversion in linux gentoo da nomorelogic
Indice da nomorelogic
DB concetti fondamentali e connessioni standard da xinyiman
Advanced Record Syntax da nomorelogic
DB concetti fondamentali e DBGrid da xinyiman
DB concetti fondamentali e TDBEdit, TDBMemo e TDBText da xinyiman
Advanced Record Syntax: un esempio pratico da nomorelogic
Superclasse form base per programmi gestionali (e non) da nomorelogic
Superclasse form base per programmi gestionali (e non) #2 - log, exception call stack, application toolbox da nomorelogic
Superclasse form base per programmi gestionali (e non) #3 - traduzione delle form da nomorelogic
Superclasse form base per programmi gestionali (e non) #4 - wait animation da nomorelogic
Un dialog per la connessione al database:TfmSimpleDbConnectionDialog da nomorelogic
Installare lazarus su mac osx sierra da xinyiman
Utenti
Stats
  • Post in totale: 11440
  • Topic in totale: 1425
  • Online Today: 20
  • Online Ever: 74
  • (Luglio 09, 2012, 11:05:53 am)
Utenti Online
Users: 0
Guests: 13
Total: 13

Disclaimer:

Questo blog non rappresenta una testata giornalistica poiché viene aggiornato senza alcuna periodicità. Non può pertanto considerarsi un prodotto editoriale ai sensi della legge n. 62/2001.