Hypercell ein ] Hypercell aus ] Zeige Navigation ] Verstecke Navigation ]
c++.net  
   

Die mobilen Seiten von c++.net:
https://m.c-plusplus.net

  
C++ Forum :: FAQ - C (C89, C99 und C11) ::  Runden     Zeige alle Beiträge auf einer Seite Auf Beitrag antworten
Autor Nachricht
Shade Of Mine
Moderator

Benutzerprofil
Anmeldungsdatum: 04.05.2001
Beiträge: 20206
Beitrag Shade Of Mine Moderator 15:43:00 02.05.2002   Titel:   Runden            Zitieren

Daniel E. hat folgende Funktion gepostet um Zahlen zu Runden:

C++:
int d2i(double d)
{
  return d<0?d-.5:d+.5;
}


Das ist sehr kompakt und bedient sich dem Trick, dass bei double to int alle nachkommastellen einfach abgeschnitten werden!

Wenn wir nun
10.3
auf Ganze runden wollen, dann machen wir das so:

10.3 + 0.5 == 10.8
Nachkommastellen weg: 10
und das passt.

und jetzt bei 10.9
10.9 + 0.5 == 11.4
Nachkommastellen weg: 11
passt auch!

Jetzt noch bei negativen Zahlen:
-5.7 - 0.5 == - 6.2
Nachkommastellen weg: -6
passt auch

Dies ist aber eine eecht 'dirty' version.
Viel eleganter kann man es machen, indem man sich der standard bibliothek bedient:

C++:
1
2
3
4
5
6
7
8
9
10
double Round(double Zahl, unsigned int Stellen)
{
    Zahl *= pow(10, Stellen);
    if (Zahl >= 0)
        floor(Zahl + 0.5)
    else
        ceil(Zahl - 0.5);
    Zahl /= pow(10, Stellen);
    return Zahl;
}


der Vorteil hierbei ist, dass man angeben kann auf wieviel Stellen man runden will.

pow() dient zum berechnen einer Potenz (include math.h)
Der Trick der hier angewendet wird ist einfacher als bei der 1. Loesung.

durch
Zahl*=pow(10,stellen)
'verschiebt' man die Stellen einfach um <stellen> nach 'rechts'
danach rundet man aehnlich wie beim 1. Beispiel und am ende werden die Stellen wieder 'zurueck geschoben'

alles in allem etwas mehr aufwand, dafuer eine schoenere und funktionalere Loesung.


ReneGs Loesung, fuer Leute denen die oben genannten zu primitiv sind.
Die Folgenden Codes beschreiben die Umformung der original Funktion in eine hochoptimierte:
C++:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
double Round(double Zahl, unsigned int Stellen)
{
    Zahl *= pow(10, Stellen);
    if (Zahl >= 0)
        floor(Zahl + 0.5)
    else
        ceil(Zahl - 0.5);
    Zahl /= pow(10, Stellen);
    return Zahl;
}
// 1. Änderung ohne Kommentar
double Round(double Zahl, unsigned int Stellen)
{
    Zahl *= pow(10, Stellen);
    if (Zahl >= 0)
        Zahl = floor(Zahl + 0.5)
    else
        Zahl = ceil(Zahl - 0.5);
    Zahl /= pow(10, Stellen);
    return Zahl;
}
// 2. Änderung: Einführen der Potenzgesetze
double Round(double Zahl, int Stellen)
{
    Zahl *= pow( 10, Stellen);
    if (Zahl >= 0)
        Zahl = floor(Zahl + 0.5)
    else
        Zahl = ceil(Zahl - 0.5);
    Zahl *= pow(10, -Stellen);
    return Zahl;
}
// 3. Änderung: Rundung überarbeitet
double Round(double Zahl, int Stellen)
{
    Zahl *= pow( 10, Stellen);
    Zahl = floor(Zahl + 0.5)
    Zahl *= pow(10, -Stellen);
    return Zahl;
}
// 4. Änderung: Zusammenfassen
double Round(double Zahl, int Stellen)
{
    return floor(Zahl * pow( 10, Stellen) + 0.5) * pow(10, -Stellen);
}
// 5. Änderung: Laufzeitverhalten verbessern
double Round(double Zahl, int Stellen)
{
    double v[] = { 1, 10, 1e2, 1e3, 1e4, 1e5, 1e6, 1e7, 1e8 };  // mgl. verlängern
    return floor(Zahl * v[Stellen] + 0.5) / v[Stellen];
}


[ Dieser Beitrag wurde am 20.06.2003 um 17:33 Uhr von Shade Of Mine editiert. ]

_________________
A language that doesn't affect the way you think about programming is not worth knowing.
Those who would give up essential Liberty, to purchase a little temporary Safety, deserve neither Liberty nor Safety.
Werbeunterbrechung
C++ Forum :: FAQ - C (C89, C99 und C11) ::  Runden   Auf Beitrag antworten

Zeige alle Beiträge auf einer Seite




Nächstes Thema anzeigen
Vorheriges Thema anzeigen
Sie können keine Beiträge in dieses Forum schreiben.
Sie können auf Beiträge in diesem Forum nicht antworten.
Sie können Ihre Beiträge in diesem Forum nicht bearbeiten.
Sie können Ihre Beiträge in diesem Forum nicht löschen.
Sie können an Umfragen in diesem Forum nicht mitmachen.

Powered by phpBB © 2001, 2002 phpBB Group :: FI Theme

c++.net ist Teilnehmer des Partnerprogramms von Amazon Europe S.à.r.l. und Partner des Werbeprogramms, das zur Bereitstellung eines Mediums für Websites konzipiert wurde, mittels dessen durch die Platzierung von Werbeanzeigen und Links zu amazon.de Werbekostenerstattung verdient werden kann.

Die Vervielfältigung der auf den Seiten www.c-plusplus.de, www.c-plusplus.info und www.c-plusplus.net enthaltenen Informationen ohne eine schriftliche Genehmigung des Seitenbetreibers ist untersagt (vgl. §4 Urheberrechtsgesetz). Die Nutzung und Änderung der vorgestellten Strukturen und Verfahren in privaten und kommerziellen Softwareanwendungen ist ausdrücklich erlaubt, soweit keine Rechte Dritter verletzt werden. Der Seitenbetreiber übernimmt keine Gewähr für die Funktion einzelner Beiträge oder Programmfragmente, insbesondere übernimmt er keine Haftung für eventuelle aus dem Gebrauch entstehenden Folgeschäden.