AGB  ·  Datenschutz  ·  Impressum  







Anmelden
Nützliche Links
Registrieren
Zurück Delphi-PRAXiS Programmierung allgemein Datenbanken Delphi Chr Feld numerisch sortieren
Thema durchsuchen
Ansicht
Themen-Optionen

Chr Feld numerisch sortieren

Ein Thema von Walter Landwehr · begonnen am 12. Aug 2015 · letzter Beitrag vom 13. Aug 2015
Antwort Antwort
Walter Landwehr

Registriert seit: 28. Mär 2006
Ort: 32816 Schieder-Schwalenberg
397 Beiträge
 
Delphi 10.4 Sydney
 
#1

Chr Feld numerisch sortieren

  Alt 12. Aug 2015, 17:15
Datenbank: Firebird • Version: 2.5.4 • Zugriff über: IBO
Hallo,
kennt jemand eine Routine in Delphi oder SQL die ein VarChr Feld numerisch sortiert.
Beispiel:

1
10
21.2
35.1
4
2
2a
...


Das Ergebnis sollte dann so aussehen

1
2
2a
4
10
21.2
35.1
...

ich habe in der Tabelle eigens ein Dezimalfeld = 'Sortierung' für die Sortierung. Der zu sortierende Wert steht im Feld Ziffer.
Walter Landwehr
Mfg

Walter
  Mit Zitat antworten Zitat
Benutzerbild von dataspider
dataspider

Registriert seit: 9. Nov 2003
Ort: 04539 Groitzsch
1.351 Beiträge
 
Delphi 11 Alexandria
 
#2

AW: Chr Feld numerisch sortieren

  Alt 12. Aug 2015, 17:39
Da du auch Buchstaben im Wert hast, dürfte es mit SQL "schwer" sein.

Du willst ja nach dem Zahlenanteil sortieren, und den musst du dann erst mal extrahieren.

Ich würde eine SP schreiben, die dir den Dezimalwert zurück liefert.

Dann kann man mit
order by (select result from SP_STRING_AS_DECIMAL(WERT))
sortieren.

So eine SP ist ja easy in wenigen Minuten geschrieben...

SP könnte so aussehen:
Code:
SET TERM ^ ;

create or alter procedure sp_string_as_decimal (
    wert varchar(20))
returns (
    result numeric(15,2))
as
declare variable laenge smallint;
declare variable pos smallint;
declare variable zeichen char(1);
declare variable temp varchar(20);
begin
  result = null;
  laenge = char_length(:wert);
  if (:laenge > 0) then
  begin
    pos = 1;
    while (:pos <= :laenge) do
    begin
      zeichen = substring(:wert from :pos for 1);
      if (:zeichen in ('0','1','2','3','4','5','6','7','8','9', '.')) then
        temp = coalesce(:temp, '')||:zeichen;
      else
        break;
      pos = :pos + 1;
    end
  end
  if (coalesce(:temp, '') <> '') then
    result = cast(:temp as numeric(15, 2));
  suspend;
end^

SET TERM ; ^

Frank
Frank Reim

Geändert von dataspider (12. Aug 2015 um 17:56 Uhr)
  Mit Zitat antworten Zitat
Walter Landwehr

Registriert seit: 28. Mär 2006
Ort: 32816 Schieder-Schwalenberg
397 Beiträge
 
Delphi 10.4 Sydney
 
#3

AW: Chr Feld numerisch sortieren

  Alt 12. Aug 2015, 18:34
Hallo Frank,

fast perfekt nur die Nachkomma stimmen noch nicht.

14.1
14.2
14.10

wird falsch sortiert, nämlich so:
14.1
14.10
14.2

sollte aber wie oben sortiert sein.
Walter Landwehr
Mfg

Walter
  Mit Zitat antworten Zitat
TBx
(Administrator)

Registriert seit: 13. Jul 2005
Ort: Stadthagen
1.893 Beiträge
 
Delphi 12 Athens
 
#4

AW: Chr Feld numerisch sortieren

  Alt 12. Aug 2015, 19:27
Das liegt daran, dass dezimal 1.1 und 1.10 identisch ist.
Will man den Teil vor und nach dem Punkt wie im Beispiel sortieren, so muss man zwei Felder daraus machen und diese dann sortieren.

Wie ist denn die Zusammensetzung der Werte definiert? Ist der Punkt der Trenner und die einzelnen Zahlen können durch Buchstaben ergänzt werden?

Sieht mir nach Abrechnungsziffern aus.
Thomas Breitkreuz
  Mit Zitat antworten Zitat
Benutzerbild von dataspider
dataspider

Registriert seit: 9. Nov 2003
Ort: 04539 Groitzsch
1.351 Beiträge
 
Delphi 11 Alexandria
 
#5

AW: Chr Feld numerisch sortieren

  Alt 12. Aug 2015, 20:35
Ja, so sieht das natürlich anders aus.
Du musst gedanklich einen sortierbaren String basteln.

Wenn also nach dem Zahlenwert vor dem Punkt und dann nach den String nach dem Punkt sortiert werden soll,
kann man den Wert vor dem Punkt links auffüllen, den Wert nach dem Punkt übernehmen.

somit kann aus 14.1 0000000014.1
und aus 14.10 0000000014.10 werden.

Die neue SP:

[EDIT] so ist korrigiert...[/EDIT]

Code:
SET TERM ^ ;

create or alter procedure sp_sortstring (
    wert varchar(20))
returns (
    result varchar(30))
as
declare variable laenge smallint;
declare variable pos smallint;
declare variable zeichen char(1);
declare variable temp varchar(20);
declare variable s1 varchar(10);
begin
  result = null;
  laenge = char_length(:wert);
  if (:laenge > 0) then
  begin
    pos = 1;
    while (:pos <= :laenge) do
    begin
      zeichen = substring(:wert from :pos for 1);
      if (:zeichen in ('0','1','2','3','4','5','6','7','8','9', '.')) then
        temp = coalesce(:temp, '')||:zeichen;
      else
        break;
      pos = :pos + 1;
    end
  end
  if (coalesce(:temp, '') <> '') then
  begin
    -- Zahl vor Punkt auffüllen
    pos = position('.', :temp);
    if (:pos = 0) then
      result = lpad(:temp, 10, '0');
    else
      result = lpad(substring(:temp from 1 for :pos -1), 10, '0')||'.'||lpad(substring(:temp from :pos + 1 for :laenge), 10, '0');
  end
  suspend;
end^

SET TERM ; ^
Ob die 10 in der SP für lpad reicht, musst du wissen.
Anpassen ist ja kein Problem.
Musst im Select natürlich den geänderten SP-Name benutzen...

Musst du öfters viele 1000 Datensätze sortieren, kann das natürlich langsam werden.
Dann sollte man darüber nachdenken, ein indexiertes Feld mitzuführen, welches in einem Trigger (Bevor Insert + Update) aktualisiert wird.

Frank
Frank Reim

Geändert von dataspider (12. Aug 2015 um 20:51 Uhr)
  Mit Zitat antworten Zitat
Walter Landwehr

Registriert seit: 28. Mär 2006
Ort: 32816 Schieder-Schwalenberg
397 Beiträge
 
Delphi 10.4 Sydney
 
#6

AW: Chr Feld numerisch sortieren

  Alt 13. Aug 2015, 09:05
Hallo Frank,

ein indiziertes Feld habe ich sowieso angelegt.

Leider liefert die SP bei

2a und 2b den gleichen Wert, damit kann die Sortierung fehlerhaft werden.

2b
2a

richtig wäre aber

2a
2b

hast Du hierzu noch eine Idee.

Ansonsten wäre die SP perfekt.
Walter Landwehr
Mfg

Walter
  Mit Zitat antworten Zitat
TBx
(Administrator)

Registriert seit: 13. Jul 2005
Ort: Stadthagen
1.893 Beiträge
 
Delphi 12 Athens
 
#7

AW: Chr Feld numerisch sortieren

  Alt 13. Aug 2015, 10:12
Davon ausgehend, dass Buchstaben immer nur ein Buchstabe hinten angehängt sein kann, kann man diesen beim ermitteln des "Dezimalanteils" ausschneiden und an das Result wieder ankleben. Dann wäre das in der Sortierung sauber mit drin.

Code:
SET TERM ^ ;

create or alter procedure sp_sortstring (
    wert varchar(20))
returns (
    result varchar(30))
as
declare variable laenge smallint;
declare variable pos smallint;
declare variable zeichen char(1);
declare variable temp varchar(20);
declare variable s1 varchar(10);
declare variable s2 varchar(10);
begin
  s2='';
  result = null;
  laenge = char_length(:wert);
  if (:laenge > 0) then
  begin
    pos = 1;
    while (:pos <= :laenge) do
    begin
      zeichen = substring(:wert from :pos for 1);
      if (:zeichen in ('0','1','2','3','4','5','6','7','8','9', '.')) then
        temp = coalesce(:temp, '')||:zeichen;
      else
        s2 = :zeichen;
        break;
      pos = :pos + 1;
    end
  end
  if (coalesce(:temp, '') <> '') then
  begin
    -- Zahl vor Punkt auffüllen
    pos = position('.', :temp);
    if (:pos = 0) then
      result = lpad(:temp, 10, '0');
    else
      result = lpad(substring(:temp from 1 for :pos -1), 10, '0')||'.'||lpad(substring(:temp from :pos + 1 for :laenge), 10, '0');
  end
  result = :result || :s2;
  suspend;
end^

SET TERM ; ^
Das ist jetzt nur mal auf die Schnelle eingeflickt, ungetestet.
Thomas Breitkreuz
  Mit Zitat antworten Zitat
Walter Landwehr

Registriert seit: 28. Mär 2006
Ort: 32816 Schieder-Schwalenberg
397 Beiträge
 
Delphi 10.4 Sydney
 
#8

AW: Chr Feld numerisch sortieren

  Alt 13. Aug 2015, 10:37
Hallo Herr Breitkreuz,

Das ist nun völlig falsch, Sortierung ergibt:
1
20.2
20.3
2a
2b
33.1
5
9.1
12.1
12.10
12.7

richtig wäre:
1
2a
2b
5
12.1
12.7
12.10
20.2
20.3
33.1
Walter Landwehr
Mfg

Walter
  Mit Zitat antworten Zitat
Benutzerbild von dataspider
dataspider

Registriert seit: 9. Nov 2003
Ort: 04539 Groitzsch
1.351 Beiträge
 
Delphi 11 Alexandria
 
#9

AW: Chr Feld numerisch sortieren

  Alt 13. Aug 2015, 13:54
Dann muss der Rest hinten ran:

[EDIT]SP - Code noch mal geändert[/EDIT]

Code:
SET TERM ^ ;

create or alter procedure sp_sortstring (
    wert varchar(20))
returns (
    result varchar(30))
as
declare variable laenge smallint;
declare variable pos smallint;
declare variable zeichen char(1);
declare variable temp varchar(20);
declare variable s1 varchar(10);
declare variable rest varchar(10);
begin
  result = null;
  laenge = char_length(:wert);
  if (:laenge > 0) then
  begin
    pos = 1;
    while (:pos <= :laenge) do
    begin
      zeichen = substring(:wert from :pos for 1);
      if (:zeichen in ('0','1','2','3','4','5','6','7','8','9', '.')) then
        temp = coalesce(:temp, '')||:zeichen;
      else
      begin
        -- Rest hinten
        rest = substring(:wert from :pos for 10);
        break;
      end
      pos = :pos + 1;
    end
  end
  if (coalesce(:temp, '') <> '') then
  begin
    -- Zahl vor Punkt auffüllen
    pos = position('.', :temp);
    if (:pos = 0) then
      result = lpad(:temp, 10, '0');
    else
      result = lpad(substring(:temp from 1 for :pos -1), 10, '0')||'.'||
               lpad(substring(:temp from :pos + 1 for :laenge), 10, '0');
    result = :result||coalesce(:rest, '');
  end
  suspend;
end^

SET TERM ; ^
Wenn dann allerdings weitere zahlen kommen, dann sollte man wirklich über Alternativen nachdenken.
Die Länge der Varchars musst du noch prüfen. Ich weiß ja nicht, wie lang die Werte bei dir werden können.

Ach ja, wenn 2a vor 2.10a kommt (siehe Anhang), dann musst du wie folgt ändern:

Code:
  ...
  if (coalesce(:temp, '') <> '') then
  begin
    -- Zahl vor Punkt auffüllen
    pos = position('.', :temp);
    if (:pos = 0) then
      result = lpad(:temp, 10, '0')||'.'||lpad('0', 10, '0');
    else
      result = lpad(substring(:temp from 1 for :pos -1), 10, '0')||'.'||
               lpad(substring(:temp from :pos + 1 for :laenge), 10, '0');
    result = :result||coalesce(:rest, '');
  end

Frank
Miniaturansicht angehängter Grafiken
sortierung.jpg  
Frank Reim

Geändert von dataspider (13. Aug 2015 um 14:12 Uhr)
  Mit Zitat antworten Zitat
Walter Landwehr

Registriert seit: 28. Mär 2006
Ort: 32816 Schieder-Schwalenberg
397 Beiträge
 
Delphi 10.4 Sydney
 
#10

AW: Chr Feld numerisch sortieren

  Alt 13. Aug 2015, 16:33
Hallo Frank,

Danke das sieht perfekt aus.
Walter Landwehr
Mfg

Walter
  Mit Zitat antworten Zitat
Antwort Antwort


Forumregeln

Es ist dir nicht erlaubt, neue Themen zu verfassen.
Es ist dir nicht erlaubt, auf Beiträge zu antworten.
Es ist dir nicht erlaubt, Anhänge hochzuladen.
Es ist dir nicht erlaubt, deine Beiträge zu bearbeiten.

BB-Code ist an.
Smileys sind an.
[IMG] Code ist an.
HTML-Code ist aus.
Trackbacks are an
Pingbacks are an
Refbacks are aus

Gehe zu:

Impressum · AGB · Datenschutz · Nach oben
Alle Zeitangaben in WEZ +1. Es ist jetzt 09:21 Uhr.
Powered by vBulletin® Copyright ©2000 - 2024, Jelsoft Enterprises Ltd.
LinkBacks Enabled by vBSEO © 2011, Crawlability, Inc.
Delphi-PRAXiS (c) 2002 - 2023 by Daniel R. Wolf, 2024 by Thomas Breitkreuz