- 3D-Druck Einstieg und Tipps         
Seite 2 von 2 ErsteErste 12
Ergebnis 11 bis 17 von 17

Thema: int Variable, Grenzbereiche

  1. #11
    HaWe
    Gast
    Anzeige

    E-Bike
    Zitat Zitat von Sisor Beitrag anzeigen
    Sicher? Sollte 2 sein, nicht 3.
    ja, stimmt ntl!

    Code:
    void setup() {
      Serial.begin(115200);
      delay(1000);
      Serial.println();
    
      
      int x=3;
      int y=2;
      int z=2;
      
      int Bsp5 = 3/2*2;
      Serial.println(Bsp5);
      
      Bsp5 = x/y*z;
      Serial.println(Bsp5);
      
    }
    
    void loop() {
      
    
    }
    output:

    2
    2
    edit: sollte durchgestrichen sein, aber Durchstreichfunktion existiert hier nicht: [s]da es alles Konstanten sind, wird die "Berechnung" evtl vom Preprozessor erledigt, ohne Compiler[/s]

    Aber auch mit volatile int kommt erwartungsgemäß dasselbe Ergebnis.

    Code:
    void setup() {
      Serial.begin(115200);
      delay(1000);
      Serial.println();
    
      
      volatile int x=3;
      volatile int y=2;
      volatile int z=2;
      
      volatile int Bsp5 = 3/2*2;
      Serial.println(Bsp5);
      
      Bsp5 = x/y*z;
      Serial.println(Bsp5);
      
    }
    
    void loop() {
      
    
    }
    Geändert von HaWe (10.05.2020 um 15:57 Uhr)

  2. #12
    Erfahrener Benutzer Roboter-Spezialist
    Registriert seit
    13.01.2014
    Beiträge
    454
    Blog-Einträge
    3
    Der Präprozessor rechnet nicht. Er bearbeitet Anweisungen zum Einfügen von Quelltext (#include), zum Ersetzen von Makros (#define), und bedingter Übersetzung (#if).
    Code:
    #define CELSIUS_ZU_FAHRENHEIT( t ) ( ( t ) * 1.8 + 32 )
    Das Makro CELSIUS_ZU_FAHRENHEIT beschreibt die Umrechnung einer Temperatur (angegeben als Parameter t) aus der Celsius- in die Fahrenheit-Skala. Auch ein Makro mit Parametern wird im Quelltext ersetzt:

    Code:
    int fahrenheit, celsius = 10;
    fahrenheit = CELSIUS_ZU_FAHRENHEIT( celsius + 5 );
    wird durch den C-Präprozessor ersetzt zu:
    Code:
    int fahrenheit, celsius = 10;
    fahrenheit = ( ( celsius + 5 ) * 1.8 + 32 );
    Quelle
    Danach darf der Compiler gemäß der spezifizierten Rechenregeln weitermachen.

  3. #13
    HaWe
    Gast
    Zitat Zitat von Sisor Beitrag anzeigen
    Der Präprozessor rechnet nicht.
    danke, da war ich einem Irrtum aufgesessen, ich dachte wirklich dass er bereits bei konstanten Ausdrücken wie
    3/2*2
    quasi das konstante Zwischenergebnis einsetzt - tut also doch immer erst hinterher der Compiler.

  4. #14
    Erfahrener Benutzer Roboter-Spezialist
    Registriert seit
    13.01.2014
    Beiträge
    454
    Blog-Einträge
    3
    Zitat Zitat von Holomino Beitrag anzeigen
    Vorsicht!
    Die Auflösung des Literals wird vom Precompiler übernommen.
    Für das Fragment:
    int main (void)
    {

    int8_t test = 3.1415/2 *2;
    if (test == 3)
    SetLED2();

    bekomme ich vom Compiler im AVR-GCC weder eine Warnung, noch einen Fehler. Im Listing sehe ich aber:

    000028b6 <main>:
    static inline void SetLED2()
    {
    PORTD.OUTSET = 0x02; }
    28b6: c0 e6 ldi r28, 0x60 ; 96
    28b8: d6 e0 ldi r29, 0x06 ; 6
    28ba: 12 e0 ldi r17, 0x02 ; 2
    28bc: 1d 83 std Y+5, r17 ; 0x05

    Da wird also weder eine Variable angelegt, noch im ganzzahligen Bereich, wie auf der eigentlichen Maschine, gerechnet.
    Insofern sind die oben angegebenen Beispielzeilen nicht wirklich tauglich, die Sache zu durchschauen.

    Besser (so, wie erwartet) wird's mit:
    int main (void)
    {
    int8_t test = 3.1415;
    test/=2;
    test*=2;
    if (test == 3)
    SetLED2();

    Hier castet sich der fraktale Anteil im Precompiler weg:
    test = 3.1415 ->3
    test /= 2 ->1
    test *= 2 ->2

    Es wird vom Compiler im Listing zwar immer noch keine Variable angelegt, aber der Aufruf von SetLED2() kommt auch nicht mehr.
    Wenn du mit Precompiler Präprozessor meinst, der rechnet nicht. GCC hat keinen Precompiler für C/C++. Die Auswertung der Literale erfolgt im Compiler.
    Bei int t = 3.1415/2 *2 ist 3.1415 ein float und 2 ein int, damit ist das Ergebnis aus 3.1415/2 ein float, damit ist das 3.1415/2 * 2 auch ein float. Das wird dann zum int (3) gecastet.
    Bei int u = 3/2 *2 sind 3 und 2 int, damit ist 3/2 ein int (1), damit ist 1*2 ein int (2). Das braucht nicht mehr gecastet werden.

    Regeln-fuer-Konvertierungen-und-Casts

  5. #15
    Erfahrener Benutzer Robotik Einstein
    Registriert seit
    18.03.2018
    Beiträge
    2.650
    Zitat Zitat von frabe Beitrag anzeigen
    Datentypen, ein spannendes und wirklich wichtiges Thema.
    So dumm war meine obige Frage gar nicht...
    Das es je nach Plattform unterschiedliche Ausprägungen gibt, beantwortet auch, warum so wenig dieser Grenzüberschreitungen im Netz steht.

    Im aktuellen Fall habe ich die Unsicherheit mit typecast beseitigt.
    Code:
    ...
    float PWMschritt = PWMbereich / FadeZeit;
    int PWMakt = (int)(AktZeit * PWMschritt + 0.5);
    ...
    Vielleicht kehre ich aber auch wieder zu <stdint.h> bzw. <inttypes.h> Datentypen zurück.
    Fand ich als Anfänger aber etwas unübersichtlicher (unleserlicher).

    DANKE für eure Inspirationen!!!
    Man findet schon was zum Thema Datentypen. Überlauf, Bitbreite etc. sind erste Grundlagen der Programmierung. Wer in Assembler bzw. mit Maschinensprache anfängt MUSS die Grundlagen lernen. Inkrementierung und Dekrementierung geht i.R. mit einem Überlauf eines Registers einher, das eine bestimmte Bitbreite hat. Ein 8Bit-Register hat 8 Bit. Das sind 256 mögliche Werte, von 0 bis 255. Das bedeutet, wenn in einem Register alle Bits gesetzt sind und eine Inkrementierung auf das Register stattfindet (80x86 bspw.: INC AL), dass das Register auf "0" springt, weil ein Überlauf stattfindet. Aber das ist nicht alles. Dazu kommt, dass beim Überlauf ein Überlauf-Bit (Carry-Flag) gesetzt wird. Es gibt weitere Befehle, die dieses Bit verrechnen können. ADC (80x86) steht zum Beispiel für Addition mit Carry. Auf diese Weise lässt sich, mit wenigen Bit eines Registers, mit riesigen Zahlen rechnen, die niemals in dieses Register vollständig hinein passen würden. Später gab es mathematische Coprozzessoren mit Registern die eine wesentlich größere Bitbreite hatten.

    In einer Hochsprache ist das aber eher unwichtig. Hier geht es mehr darum, die Grenzwerte der Datentypen zu kennen und sich innerhalb dieser zu bewegen. Für eine Berechnung eines Wertes brauche ich also den passenden Datentyp, wenn es keine Überraschungen geben soll.



    MfG

  6. #16
    Erfahrener Benutzer Fleißiges Mitglied
    Registriert seit
    07.06.2019
    Beiträge
    147
    Zitat Zitat von Moppi Beitrag anzeigen
    Wertes brauche ich also den passenden Datentyp, wenn es keine Überraschungen geben soll.
    DAS ist das A und O bei allen Deklarationen und Zwischenberechnungen - das Fazit - niemals zulassen das ungereimtheiten auftreten können.
    Nur ist es nict immer so leichte, besonders wenn man bestehende Codes anpasst oder anch langer Zeit mal wieder anderweitig verwenden möchte.
    __________________________________________________ _
    | Sprache: C | Teensy 3.2 | Arduino 2.x | Status: EwigerAnfaenger |

  7. #17
    Erfahrener Benutzer Robotik Einstein
    Registriert seit
    07.03.2011
    Beiträge
    1.899
    Zitat Zitat von frabe Beitrag anzeigen
    Nur ist es nict immer so leichte, besonders wenn man bestehende Codes anpasst oder anch langer Zeit mal wieder anderweitig verwenden möchte.
    Mit ein wenig Disziplin geht das schon. Das fängt damit an, daß man unsigned Typen vermeidet. Mathematisch machen sie keinen Sinn, eigentlich sind sie Bitfelder wie SFRs oder Bitsequenzen eines Übertragungsprotokolls. Darauf sollte man keine arithmetischen Operationen sondern nur logische Operatoren anwenden. Wenn man Werte daraus im Programm weiter verwenden will, sollte man sie so schnell wie möglich in Zahlen umwandeln. Man sollte generell mit Objekten, die keine Zahlen sind, nicht rechnen. Daher ist es auch unerheblich ob chars signed oder unsigned sind. 'a' + 'b' macht keinen Sinn, es mag zwar in C ein Ergebnis geben, in anderen Sprachen eher nicht. Die einzige arithmetische Operation die auf chars Sinn macht ist '0' + [0..9]. Sie liefert das Zeichen für die Ziffern 0 bis 9 und das unabhängig davon, ob chars signed oder unsigned sind oder der Zeichensatz ASCII oder EBCDIC ist. Wenn man aus Platz oder Geschwindigkeitsproblemen 8-Bit Variable für nötig hält, verwendet man int8_t. Will man auf allen Architekturen schnell sein, sollte man int_fast8_t verwenden.

    Ansonsten ist ein int immer mindestens 16 Bit. Wenn man portabel programmiert, verlässt man sich nicht auf das Verhalten bei einem Über oder Unterlauf. Man sollte also nicht davon ausgehen, daß 32767 + 1 = -32768 ist. Wenn man das so will, sollte man es explizit so programmieren. Dann läuft das Programm auch, auf einem System mit 32 oder 64 Bit Integer. Wenn man so programmiert, kommt man sogar mit den klassischen C-Typen int, long int (32 Bit) oder long long (64 Bit) aus. Besser lesbar sind sicher die Typen aus stdint.h wie int32_t und int64_t.

    Daß das geht, sieht man an vielen Programmen. Beim Umstieg von 32 auf 64 Bit mussten sie nur neu kompiliert und gegen die 64 Bit Library gelinkt werden. Wer aber trickst, Überläufe explizit einkalkuliert, Shifts verwendet statt zu teilen, oder zu addieren statt bitweise zu odern, hat da eher Schwierigkeiten.

    MfG Klebwax
    Strom fließt auch durch krumme Drähte !

Seite 2 von 2 ErsteErste 12

Ähnliche Themen

  1. Variable auf LCD ausgeben
    Von Haveaniceday im Forum C - Programmierung (GCC u.a.)
    Antworten: 6
    Letzter Beitrag: 20.07.2017, 19:57
  2. [ERLEDIGT] C++ Hinweis: variable 'pHelp' set but not used [-Wunused-but-set-variable]
    Von geWichtig im Forum C - Programmierung (GCC u.a.)
    Antworten: 3
    Letzter Beitrag: 20.02.2017, 16:09
  3. Komplexe Variable
    Von DanielSan im Forum Basic-Programmierung (Bascom-Compiler)
    Antworten: 2
    Letzter Beitrag: 19.06.2011, 15:30
  4. if Variable > 1 then Variable + 1
    Von klaus1973 im Forum Basic-Programmierung (Bascom-Compiler)
    Antworten: 4
    Letzter Beitrag: 26.11.2005, 15:31
  5. Pin mit Variable vergleichen
    Von Kundesbanzler im Forum Basic-Programmierung (Bascom-Compiler)
    Antworten: 1
    Letzter Beitrag: 23.02.2005, 15:58

Stichworte

Berechtigungen

  • Neue Themen erstellen: Nein
  • Themen beantworten: Nein
  • Anhänge hochladen: Nein
  • Beiträge bearbeiten: Nein
  •  

12V Akku bauen