AGB  ·  Datenschutz  ·  Impressum  







Anmelden
Nützliche Links
Registrieren
Thema durchsuchen
Ansicht
Themen-Optionen

Problem mit Fraktion/Prozentrechnung

Ein Thema von DGL-luke · begonnen am 22. Mai 2005 · letzter Beitrag vom 23. Mai 2005
Antwort Antwort
Benutzerbild von DGL-luke
DGL-luke

Registriert seit: 1. Apr 2005
Ort: Bad Tölz
4.149 Beiträge
 
Delphi 2006 Professional
 
#1

Problem mit Fraktion/Prozentrechnung

  Alt 22. Mai 2005, 23:09
Ich mach gerade ne kleine Anwendung, die mir n paar statistische Sachen über eine(n) text(datei) gibt. naja, eigentlich berechnet sie zur zeit nur die verteilung der buchstaben:

Delphi-Quellcode:
procedure TForm1.BCheckClick(Sender: TObject);
var c:char;
    counts:array[0..25] of integer;
    fraction:array[0..25] of single;
    i,k:integer;
    PB:TRBProgressBar;
begin
for PB in bars do
 PB.Position:=0; //Ja, Marabu und Sakura, seht nur hin, ich weise der Schleifenvariable etwas zu ;)

for i:= 0 to length(counts)-1 do //initialize the array
 counts[i]:=0;

for i := 1 to memo1.GetTextLen do
 begin
  application.ProcessMessages; //damit das ganze nicht hängenbleibt
  c:=memo1.text[i]; //dem char wird ein buchstabe des memos zugewiesen.
  c:=ansilowercase(c)[1]; //der char wird kleingemacht. was passiert eigentlich wenn er z.b. #10 ist? das muss ich gleich noch alles überprüfen/abfangen.
  PBSum.Position:=round(i / memo1.GetTextLen * 100); //hier kommt der gesamtfortschritt raus.
  counts[ord(c)-97]:=counts[ord(c)-97]+1; //wenn c=#97='a' ist, wird counts[97-97] inkrementiert.
 end;

for i:= 0 to 25 do
 fraction[i]:=counts[i]/memo1.GetTextLen; //hier werden die anteile an der gesamtdatei berechnet.

for i:= 0 to 100 do //diese schleife ist eigentlich nur design.
 for k:= 0 to 25 do
  begin
   application.ProcessMessages; //noch einmal antifreeze :D
   if bars[k].Position < round(fraction[k]*100) then //wenn die progressbar noch nicht soweit ist wie es der anteil sagt, dann...
    bars[k].Position:=bars[k].Position + 1; //...wird position inkementiert.
  end;
end;
1. ja die performance ist schlecht. danach frag ich aber gar nicht. oder doch: wenn ich progressmessages nicht so oft aufrufe, wirkt sich das auf die performance aus, oder werden sowieso alle anstehenden messages abgearbeitet?
2. das eigentliche problem:
es funktioniert alles wunderbar, aber ich bekomm zum schluss immer nur ganz mickrige anteile raus. in der summe ergibt es nie auch nur annähernd hundert (zeilenumbrüche muss man ja rausrechnen). es bewegt sich so zwischen 60% und 80%, wobei 60% typisch ist. die linefeeds machen 2-4% aus. ich hab also irgendwo was falsch. evtl. irgendwo so gravierende rundungsfehler? die bars werden ja in der letzten schleife zugewiesen. das sieht für mich aber ganz richtig aus?!

EDIT: gerade nen testlauf gemacht mit einfach nur 'aaaaaaa...'. gibt er mir brav meine 100%. also doch n rundungsfehler?!
EDIT2: liegt es vielleicht doch an den ganzen sonderzeichen('(', '-','_' usw.) die sich in so nem textfile verstecken? hab jetzt noch n paar solche testfälle laufen gelassen, und da geht es.
Lukas Erlacher
Suche Grafiktablett. Spenden/Gebrauchtangebote willkommen.
Gotteskrieger gesucht!
For it is the chief characteristic of the religion of science that it works. - Isaac Asimov, Foundation I, Buch 1
  Mit Zitat antworten Zitat
Minz

Registriert seit: 19. Dez 2002
476 Beiträge
 
#2

Re: Problem mit Fraktion/Prozentrechnung

  Alt 23. Mai 2005, 01:54
2 Stellen kommen mir komisch vor:

Zitat:
PBSum.Position:=round(i / memo1.GetTextLen * 100);
müsste es nicht 100/memo1.GetTextLen*i; sein?

als Alternative könntest du dir das aber auch sparen, wenn du die Progressbar-Länge auf die Memo-Länge setzt, und mit jedem i, PBSum.Step machst.

Zitat:
fraction[i]:=counts[i]/memo1.GetTextLen;
Sollte der Anteil nicht eine Prozentangabe sein? Ist ja klar, dass du so eine sehr kleine Zahl bekommst, weil es immer z=x/y, y>x ist und damit z<1 ist
fraction[i]:=100/memo1.GetTextLen*counts[i];
  Mit Zitat antworten Zitat
Robert_G
(Gast)

n/a Beiträge
 
#3

Re: Problem mit Fraktion/Prozentrechnung

  Alt 23. Mai 2005, 06:57
ot:
Zitat:
PB.Position:=0; //Ja, Marabu und Sakura, seht nur hin, ich weise der Schleifenvariable etwas zu
Nope, der Zeiger der Variable ist constant, das hindert dich aber nicht daran ihre Properties zu besetzen. (Das ganze wäre auch sinnlos, wenn das nicht ginge )
  Mit Zitat antworten Zitat
Benutzerbild von jim_raynor
jim_raynor

Registriert seit: 17. Okt 2004
Ort: Berlin
1.251 Beiträge
 
Delphi 5 Standard
 
#4

Re: Problem mit Fraktion/Prozentrechnung

  Alt 23. Mai 2005, 09:35
Zitat von DGL-luke:
1. ja die performance ist schlecht. danach frag ich aber gar nicht. oder doch: wenn ich progressmessages nicht so oft aufrufe, wirkt sich das auf die performance aus, oder werden sowieso alle anstehenden messages abgearbeitet?
Ja, es wirkt sich sehr negativ aus. Mein Vorschlag: Es sollte reichen, wenn du alle 10000 Zeichen oder so mal ein ProcessMessage machst. Nimms mal raus und es wird erheblich schneller laufen
Christian Reich
Schaut euch mein X-COM Remake X-Force: Fight For Destiny ( http://www.xforce-online.de ) an.
  Mit Zitat antworten Zitat
shmia

Registriert seit: 2. Mär 2004
5.508 Beiträge
 
Delphi 5 Professional
 
#5

Re: Problem mit Fraktion/Prozentrechnung

  Alt 23. Mai 2005, 10:11
So bleibt extrem viel Performance auf der Strecke, da bei Aufruf von memo1.Text
jedesmal alle Zeichen geholt werden.
Das Property TMemo.Text mag ja aussehen, als ob es ein gewöhnlicher String wäre;
intern ist aber ein relativ hoher Aufwand nötig, diesen String zu liefern.
Delphi-Quellcode:
for i := 1 to memo1.GetTextLen do
begin
  application.ProcessMessages; //damit das ganze nicht hängenbleibt
  c:=memo1.text[i]; //dem char wird ein buchstabe des memos zugewiesen.
  c:=ansilowercase(c)[1];
deutlich schneller:
Delphi-Quellcode:
var
   s : string;
...
   s := AnsiLowerCase(memo1.Text); // Daten aus Memo in einem Rutsch holen und gleich in Kleinbuchstaben wandeln
for i :=1 to Length(s) do
begin
   c := s[i];
   ...
Andreas
  Mit Zitat antworten Zitat
Benutzerbild von DGL-luke
DGL-luke

Registriert seit: 1. Apr 2005
Ort: Bad Tölz
4.149 Beiträge
 
Delphi 2006 Professional
 
#6

Re: Problem mit Fraktion/Prozentrechnung

  Alt 23. Mai 2005, 13:40
das hört sich ja alles schon ganz vielversprechend an.

@Minz: also das zuweisen von PBSum funktioniert einwandfrei. deshalb werd ich da jetzt auch nichts ändern. fraction ist der anteil aller vorkommen eines zeichens und sollte im regelfall kleiner 1 sein. wenn fraction[65] gleich 0,5 ist, dann besteht exakt die hälfte der textdatei aus 'A's.

@Robert_G: ja, logisch. is mir klar.

@jim_raynor: ok werd ich mal machen. ne einfache if i mod 1000 = 0 then asollte ja reichen.

@shmia: ja, das seh ich mir mal an.

@all: thx!

EDIT: damn, das war jetzt ein schub. und ich hab nur das memo1.text, memo1.gettextlen in ne variable ausgelagert und das processmessages verändert, also nicht so oft aufgerufen. mehr performance brauch ich glaub ich gar nicht. 92 kB in weniger als einer Sekunde. zuvor hab ich mehr als eine Minute gebraucht.
Lukas Erlacher
Suche Grafiktablett. Spenden/Gebrauchtangebote willkommen.
Gotteskrieger gesucht!
For it is the chief characteristic of the religion of science that it works. - Isaac Asimov, Foundation I, Buch 1
  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 22:59 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 by Thomas Breitkreuz