Delphi-PRAXiS

Delphi-PRAXiS (https://www.delphipraxis.net/forum.php)
-   Datenbanken (https://www.delphipraxis.net/15-datenbanken/)
-   -   Delphi Ist sowas mit SQL machbar? (https://www.delphipraxis.net/188436-ist-sowas-mit-sql-machbar.html)

juergen 3. Mär 2016 15:45

Datenbank: Pervasive • Version: 10 • Zugriff über: PDAC

Ist sowas mit SQL machbar?
 
Hallo zusammen,

ich habe eine Tabelle wo Abwesenheiten drin stehen. Pro Tag ein Datensatz.

Jetzt benötige ich folgende Ausgabe:

PersNr xBeginn-DatumUhrzeit xEnde-DatumUhrzeit xFehlart x
000000011602.06.201600:00:0005.06.201624:00:00Urlaub
000000011704.06.201600:00:0014.06.201624:00:00Urlaub

Die Felder mit dem x habe ich.
Der Knackpunkt sind das Beginn-Datum und das Ende-Datum!
Es gibt in der Tabelle ja pro Tag einen Datensatz.

Ich wüßte jetzt nicht ob
a) das überhaupt möglich ist und
b) wie man das lösen könnte

Für eure Hilfe bedanke ich mich schon mal vorab!

4dk2 3. Mär 2016 15:52

AW: Ist sowas mit SQL machbar?
 
hmm,
Möglichkeit A wäre wenn die Datenbank unterstützt ne Stored Procedure :-D
Möglichkeit B wären Subquerys, je nachdem welche Datenbank :)

EDIT:
Streichen:
Um welche Datenbank geht es ? :lol:

War zu schnell XD

DeddyH 3. Mär 2016 15:53

AW: Ist sowas mit SQL machbar?
 
Pervasive, steht oben im Eingangspost ;)

4dk2 3. Mär 2016 15:57

AW: Ist sowas mit SQL machbar?
 
Zitat:

Zitat von DeddyH (Beitrag 1331975)
Pervasive, steht oben im Eingangspost ;)

ja ich habs nachher gesehen... das Brett vorm Kopf hatte es verdeckt ;)

Neutral General 3. Mär 2016 16:09

AW: Ist sowas mit SQL machbar?
 
Das einfachste wäre wahrscheinlich wie schon genannt eine Stored Procedure.
Da kannst du dann alle Datensätze selektieren, entsprechend nach Personalnummer und Datum sortieren und jedes mal wenn sich die Art der Abwesenheit oder die Personalnummer ändert, oder wenn der Aktuelle Tag > LetzterTag + 1 erzeugst du einen Datensatz.

Jumpy 3. Mär 2016 16:23

AW: Ist sowas mit SQL machbar?
 
Ich behaupte einfach mal, dass das auch nur mit SQL geht, aber wahrsch. kompliziert.
Idee:
Left Join der Tabelle auf sich selber, Bedingung PNR=PNR und Datum=Datum-1
Weiterer Left Join der Tabelle auf sich selber, Bedingung PNR=PNR und Datum=Datum+1

Die Datumse wo einer der Joins null ist ist Begin oder Ende eines Fehlzeitraumes.
Die Datumse wo keiner der Joins null ist inmitten eines Fehlzeitraumes. Wird nicht weiter beachtet.
Die Datumse wo beide Joins null sind, sind 1-tägige Fehlzeiten.

Darüber ein weiterer Select der (bei mehrtägigen FZ) einem Begin-Fehlzeit-Datum das nächst kleinere Ende-Fehlzeit-Datum zuordnet.

Stevie 3. Mär 2016 16:27

AW: Ist sowas mit SQL machbar?
 
Das hier sieht aus, als ob es für dich hilfreich sein könnte, ob das mit deiner DB geht, musst du selbst schauen: http://www.sqlservercentral.com/articles/T-SQL/71550/

himitsu 3. Mär 2016 17:14

AW: Ist sowas mit SQL machbar?
 
Kann man nicht über 'ne WindowFunction, oder so, den aufeinanderfolgenden zusammengehörigen Datensätzen eine gleiche ID vergeben (wenn davor die gleiche Fehlart, dann gibt mir die selbe ID)
und das zusammengehörige dann per GROUP BY und paar AggregateFunctions zusammenfassen?

jobo 3. Mär 2016 18:16

AW: Ist sowas mit SQL machbar?
 
Zitat:

Zitat von juergen (Beitrag 1331972)
Ich wüßte jetzt nicht ob
a) das überhaupt möglich ist und
b) wie man das lösen könnte

Ich versteh glaub ich das Problem nicht ganz. Du kannst das so machen, die Tabelle wird es Dir nicht übel nehmen.

Aber was soll dadurch erreicht werden?

Werden die "täglichen" Datensätze so generiert?
Ist das eine Erweiterung die tägliche DS sparen soll?

Ich vermute, es geht darum für 10 Tage Urlaub 9 Datensätze zu sparen oder?

Fakt ist, die Spalten mit Beginn / Ende zusammen mit der Vorgabe ergeben redundante Daten. Das ist nicht gut gemäß Vorschrift und Erfahrung.

juergen 3. Mär 2016 19:00

AW: Ist sowas mit SQL machbar?
 
Hallo zusammen,

erst einmal Danke für die ganzen Antworten!

@jobo,
Die DB (bzw. Tabelle) ist so gegeben, da kann nichts angepasst werden. Der Kunde erwartet als Ergebnis nicht die einzelnen Tages-Datensätze sondern das Ganze zusammengefasst in einer Zeile, wenn es der gleiche Fehlgrund ist.
Als Ergebnis soll eine csv-Datei "rauskommen" die der Kunde in einem anderen System wieder importiert.
Ich werde das Ganze nun mit Delphi lösen. Das scheint erst mal relativ einfach, wenn die Datenmenge nach PersNr und Datum sortiert ist.

Allen ein erholsamer Feierabend! :dp:

Jumpy 4. Mär 2016 08:01

AW: Ist sowas mit SQL machbar?
 
Ich habe das auch schon mal gebraucht bei einer Schnittstelle zw. einem Zeitwirtschaft-System auf der einen und einem Personalabrechnungssystem auf der anderen Seite und da die in #6 beschriebene Lösung verwendet. Da das in meiner alten Firma war, hab ich da aber keinen Zugriff mehr drauf, sonst hätte ich das im Detail/SQL gepostet, statt nur grob den Weg/die Idee.

Mit dem reinen SQL-Statement konnte ich auf ein eigenes Delphi-Pogramm verzichten und eine Standard-Export-Lösung der Software verwenden (die nur mit einem SQL-Statement gefüttert werden musste).

jobo 4. Mär 2016 08:22

AW: Ist sowas mit SQL machbar?
 
@juergen: Auch wenn es angeblich einen Datensatz pro Tag gibt in der Tabelle, der Tag selbst, das Datum, steht nicht drin? Ist zumindest nicht als vorhanden markiert.

Sinnvoll bzw. bequem wären schon Window Functions, aber pervasive kenne ich nicht, für mich etwas exotisch, weiß nicht, ob die das kann. Ebenso SP, hier würde ein bequemer Ansatz wahrscheinlich recht ähnlich zu einem Vorgehen in Delphi, dursteppen und Liste ausspucken.

Wenn es nicht um große Datenmengen geht und die Lösung auch nicht für mehrere andere Systeme zur Verfügung stehen muss, ist eine Lösung per Client sicher am einfachsten. Ansonsten müsste man sich mal anschauen, was die eingesetze pervasive Version so alles kann, es geht bestimmt, auch ohne Windows Functions.

juergen 4. Mär 2016 18:02

AW: Ist sowas mit SQL machbar?
 
@jobo,
erst mal danke für deinen Einsatz!
Inzw. habe ich das ganze mit Delphi gelöst, indem ich die Query.Fields jeweils in ein Record schaufle und dann die Datensätze vergleiche. Wenn vom vorherigen Datensatz die PersNr, der Fehlgrund und das Datum + 1Tag(!) gleich ist, dann mache ich nichts ansonsten schreibe ich das ganze in eine Datei.
Ich hatte mit mehreren joins auf SQL-Ebene immer nur Teilergebnisse erreicht, der Aufwand wäre vermutlich sehr viel höher geworden und das ganze Statement ziemlich komplex. Weiterhin ist das Laufzeitverhalten über Pervasive mit 3 joins ziemlich mau. Mein Vergleich in Delphi geht sehr schnell (ca. 12.000 Datensätze als Ergebnis von ca. 2,3 Mio Datensätze in der Tabelle...).

Ein schönes Wochenende!

himitsu 4. Mär 2016 22:23

AW: Ist sowas mit SQL machbar?
 
Hatte erst meine Überlegungen so angefangen
SQL-Code:
SELECT PersNr, min(BeginnDatum+BeginnUhrzeit) AS Beginn, max(EndeDatum+EndeUhrzeit) AS Ende, Fehlart,
  lag(PersNr || Fehlart, ***) OVER (
    --PARTITION BY PersNr, Fehlart
    --ORDER BY EndeDatum+EndeUhrzeit
  ) AS GroupID
GROUP BY GroupID
ORDER BY PersNr, Fehlart, BeginnDatum, BeginnUhrzeit
aber das mit den WindowFunctions ist nicht so mein Ding.

Also dann doch mit den guten alten SubSelects weiterüberlegt. :stupid:
SQL-Code:
SELECT PersNr, min(BeginnDatum+BeginnUhrzeit) AS Beginn, max(EndeDatum+EndeUhrzeit) AS Ende, Fehlart,
  (
    WITH RECURSIVE Temp(Datum) AS (
      SELECT Abwesenheiten.BeginnDatum
      UNION ALL
      SELECT Loop.BeginnDatum
      FROM Abwesenheiten AS Loop
      WHERE Loop.BeginnDatum = Datum - 1
        AND Loop.PersNr = Abwesenheiten.PersNr
        AND Loop.Fehlart = Abwesenheiten.Fehlart
    )
    SELECT min(Datum) FROM Temp;
  ) AS GroupID
FROM Abwesenheiten
GROUP BY GroupID
ORDER BY PersNr, Fehlart, BeginnDatum, BeginnUhrzeit
>> nur grobe ungetestete Gedankengänge <<
Im Prinzip irgendwie den zuzammengehörigen Datensätzen eine Kennung verpassen und da ein GROUP BY drauf anwenden.

jobo 5. Mär 2016 06:00

AW: Ist sowas mit SQL machbar?
 
Zitat:

Zitat von himitsu (Beitrag 1332127)
Im Prinzip irgendwie den zuzammengehörigen Datensätzen eine Kennung verpassen und da ein GROUP BY drauf anwenden.

Ja, so könnte das gehen, wenn jürgens db rekursive CTE kann. Das bezweifel ich.
Ohne etwas Optimizer logic Richtung CTE und Window Functions möchte ich auch nicht wissen, wie die DB dann auf den Files rumrödelt.
@jürgen:
Wenn die DB da so wenig Unterstützung liefert, ist es wohl der beste Ansatz, db seitig ein Scan, Verarbeitung im Client, fertig.


Alle Zeitangaben in WEZ +1. Es ist jetzt 01:15 Uhr.

Powered by vBulletin® Copyright ©2000 - 2025, Jelsoft Enterprises Ltd.
LinkBacks Enabled by vBSEO © 2011, Crawlability, Inc.
Delphi-PRAXiS (c) 2002 - 2023 by Daniel R. Wolf, 2024-2025 by Thomas Breitkreuz