Datum: 27.06.2008 18:55
Hallo ! Versuche gerade Funktionen in ein paar Header zu verpacken. (die sich auch untereinanader einbinden) Ich suche jetzt ein paar "Regeln", da hier im Forum anscheinend nur sonderfälle (bis auf http://www.mikrocontroller.net/articles/Funktionen...) ) behandelt werden. - Modulübergreifende Funktionen müssen als Funktionsrumpf in das Header-File. - Globale Variablen werden im C-File Definiert (erstellt, Platz reserviert), und im Header-File mit "extern" Deklariert (bekannt gemacht). Ich bekomme keine Compiler-Fehlermeldungen in Richtung "redefinition", selbst wenn ein Header Mehrfach eingebunden wird. --- Wenn ich structs, enums und typedefs auslagern möchte muss ich diese bedingte Kompilierung benutzen : #ifndef _test #define _test Code #endif Gibt es da keinen eleganteren Weg wie bei der Variablen-Definition und extern Deklaration ???
Datum: 27.06.2008 18:59
Alle Funktionen, Variablen, und includs kommen in das C File Alle Definitionen, Strukturen und Definitionen ins H File. Werden die Variablen auch in anderen C Files benötigt kommen die auch mit einem extern in das H File
Datum: 27.06.2008 19:02
Und wann muss man einer Funktion ein "static" voranstellen?
Datum: 27.06.2008 19:10
Eine einfache Möglichkeit (ähnlich Deiner) ist
#ifndef _YourHeaderFileName_
#ifndef _YourHeaderFileName_
am Anfang Deiner Header-Datei und
#endif
am Ende der Header-Datei ( =: Makro-Ansatz).
Eine andere Möglichkeit ist es, Header-Dateien genau
einmal zu includen. Da es aber oft vorkommt, dass eine
bestimmte Header-Datei schon in meist mehreren anderen
includiert ist, hast Du keine andere Wahl als den
Makro-Ansatz.
Zu bemerken ist ausserdem, in fast allen
Header-Dateien der Standard-Bibliotheken genau so
verfahren wird (ausser vieleicht einigen MFC-generierten
Dateien, da bekommst Du aber auch Fehlermeldungen).
Gruss
Jörg
Datum: 27.06.2008 20:14
@ Gast Jepp, glaube so habe ich das beschrieben. Mit ging es aber um structs und typedefs -> die kann ich nicht in ein Header schreiben den ich mehrfach "include". Jedenfalls nicht ohne diese Define-Struktur. @ Andreas H. static hat was mit der Gültigkeit der Variable zu tun, sie ist sind ständig vorhanden aber nur von der dem entsprechenden Gültigkeitsbereich verfügbar. static in einer Funktion lässte also die Variable "am Leben" ihr Wert wird in den erneuten Aufruf der Funktion gerettet. Im gegensatz zu Automatischen Variablen, die werde bei jedem Funktionseintritt erneuet erstellt. (wenn ich das richtig verstanden habe) @ Jörg Hmmm, Du hast mich überzeugt ! Jetzt werde ich das wohl oder übel benutzen müssen... Aber irgendwie ist das ein bisschen merkwürdig ! Nur fürs verständnis : Ich gehe mal davon aus das es gut ist jedes Headerfile in jedes andere und auch in main.c zu includieren. (Das sollte doch Fehlerquellen die der Linker/Compiler nicht sieht erschlagen ?!?!) Jedes Modul (also Header mit dem dazugehörigen Code) wird ja seperat übersetzt. Deshalb müssen die Strukturen die gemeinsam genutzt werden bekannt sein. Unsere beiden "Makro-Kunstwerke" eleminieren den Includierten Code weg sobald er doppelt da wäre (>1 - mal includiert). Wird dieses Your_Header_FileName durch den PräProzessor ersetzt ??? Bzw. Warum zweimal dieses : #ifndef bla
Datum: 27.06.2008 21:00
Grundsätzlich solltest Du nur das includieren was Du (bzw. das entsrechende C/H-File) wirklich brauchst. Der Makro-Ansatz sorgt dann dafür, dass jede Struktur/Klasse nur ein einziges mal definiert wird. Und ich muss Dich leider ein weiteres mal entäuschen: _YourHeaderFileName_ muss von Dir jedes mal manuell durch z.B. FileName oder ModuleName ersetzt werden. Gruss Jörg
Datum: 27.06.2008 21:01
Hab ich fast vergessen: Was meinst Du mit
> Bzw. Warum zweimal dieses : #ifndef bla
#ifndef wird doch nur einmal verwendet
Gruss
Jörg
Datum: 27.06.2008 21:43
Jörg Du schriebst :
----
Eine einfache Möglichkeit (ähnlich Deiner) ist
#ifndef YourHeaderFileName
#ifndef YourHeaderFileName
am Anfang Deiner Header-Datei und
#endif
am Ende der Header-Datei ( =: Makro-Ansatz).
----
Denke mal die zweite Zeile sollte mit #define beginnen.
Dann sieht es aber genauso wie bei meinem Beispiel aus ?!
Waren denn meine restlichen Ausführungen (Anfängertechnisch-)richtig ???
Das mit allen Headers includieren kommt bei meinem aktuellen projekt
irgendwie fast zwangsläufig.
(rs232.h, timing.h, egt_map.h, defines.h, led.h, adc.h, eeprom.h --
Namen sind selbstredend)
Hab also mein main.c leergeräumt.
Datum: 27.06.2008 21:51
Jörg wrote: > Eine einfache Möglichkeit (ähnlich Deiner) ist > > #ifndef __YourHeaderFileName__ > #ifndef __YourHeaderFileName__ > > am Anfang Deiner Header-Datei und > > #endif > > am Ende der Header-Datei ( =: Makro-Ansatz). Nix da, Namen, die einen oder mehrere Unterstriche vorangestellt bekommen, sind für den Compiler/Dessen Lib reserviert. Finger weg davon als normaler Codetipper. Der gängige Standard ist:
#ifndef HEADERFILENAME_H_ #define HEADERFILENAME_H_ //... #endif /* HEADERFILENAME_H_ */ |
oder
#ifndef HEADERFILENAME_H #define HEADERFILENAME_H //... #endif /* HEADERFILENAME_H */ |
Wobei die Namen im Prinzip egal sind.# Und ja, sowas ist durchaus üblich und gängig in der C/C++ Landschaft. Nennt sich Include-Guard: http://en.wikipedia.org/wiki/Include_guard.
Datum: 27.06.2008 21:59
@ frischling,
sorry, copy-and-paste-Fehler, muss natürlich #define statt zweitem
#ifndef heissen!!!
@ Simon K.,
ich weiss zwar, dass (glaube ich) alle compilereigene header-Dateien
mit doppelten Underscore ("__BLABLA..") anfangen, wusste aber nicht,
dass es sich um einen Standard handelt. Ist das eine
betriebssystemunabhängige Regel (ich arbeite fast nur unter Windows)?
Gruss
Jörg
Datum: 27.06.2008 22:00
Trotzdem...wann muss man einer Funktion ein "static" voranstellen?
Datum: 27.06.2008 22:12
"static" bedeutet bei einer Funktion (oder einer globalen Variablen), dass nur in dieser "Compilation unit" d.h. .c-Datei darauf zugegriffen werden kann, ist also quasi das Gegenstück zu "extern". Was bringt das? Der Compiler kann "static"-Funktionen z.B. inline'n und gar nicht als aufrufbare Funktion im Programm lassen (die klassischen *_putc-Funktionen sind da gute Beispiele - wenn in so einer Funktion nur ein I/O-Register geschrieben und anschließend gewartet wird). hth. Jörg ps. Das muss man gar nicht voranstellen ;)
Datum: 27.06.2008 23:41
Jörg wrote: > ich weiss zwar, dass (glaube ich) alle compilereigene header-Dateien > mit doppelten Underscore ("__BLABLA..") anfangen, wusste aber nicht, > dass es sich um einen Standard handelt. Ist das eine > betriebssystemunabhängige Regel (ich arbeite fast nur unter Windows)? Ja. Das ist im C-Standard so definiert. Das es in C keine Namespaces gibt, wurd ein künstlicher 'Namespace' eingerichtet, der besagt, dass alle Namen die mit zwei _ beginnen oder einem _ gefolgt von einem Grossbuchstaben, für die Systembibliothek reserviert sind. Benutzte solche Namen einfach nicht und du wirst nie das Problem haben, dass dir irgendein Systemheader in deine Variablen-Definitionen hineinpfuscht.
Datum: 27.06.2008 23:42
Andreas H. wrote:
> Trotzdem...wann muss man einer Funktion ein "static" voranstellen?
Wenn du haben willst, dass der Sichtbarkeitsbereich dieser
Funktion nur auf das *.c beschränkt bleibt, in dem diese Funktion
definiert ist.
Datum: 28.06.2008 10:48
#ifndef FILE #define FILE Code #endif ^--- So müsste es doch auch Funktionieren ?! FILE wird doch vom Präprozessor mit dem Dateinamen des gerade durchlaufenden Source/Modul ersetzt.
Datum: 28.06.2008 10:49
File ohne leerzeichen sollte es heissen
Datum: 28.06.2008 10:57
frischling wrote: > FILE wird doch vom Präprozessor mit dem Dateinamen des gerade > durchlaufenden Source/Modul ersetzt. Aber nur im normalen C-Quelltext. #ifndef, #define etc. sind Präprozessoranweisung. Die lässt der Präprozessor in seiner Gesamtheit in Ruhe. Irgendwo muss ja auch mal Schluss sein mit den Ersetzungen :-)
Datum: 28.06.2008 11:27
Hmmm, schade ! Rekursive Ersetzungen wären bestimmt auch sehr interessant... Vielleicht sollte man das hier erlangte wissen in deinen Artikel mit aufnehmen. (kbuchegg) Oder gibt es einen C-Kurs in dem sowas behandelt wird ?!
Datum: 28.06.2008 13:52
frischling wrote:
> Oder gibt es einen C-Kurs in dem sowas behandelt wird ?!
In jedem C-Buch würde ich sagen ;)
Datum: 28.06.2008 14:33
@frischling: scheinbar tappt jeder mal in die Falle zu glauben,
__FILE__ |
ist in dem Fall der Filename der Header-Datei. Include fügt aber nur die Headerdatei in das C-File ein und dann wird von Präprozessor
__FILE__ |
durch den Namen des C-Files (in dem includiert wird!) ersetzt. Wünschenswert wäre ja mal ein Makro z.B. der Form
__INCLUDEFILE__ |
mit dem Du den Headername automatisch generieren kannst, gibts aber leider nicht. Gruss Jörg
Antwort schreiben
Die Angabe einer Email-Adresse ist freiwillig. Wenn Sie automatisch per Email über Antworten auf Ihren Beitrag informiert werden möchten, melden Sie sich bitte an.
Wichtige Regeln - erst lesen, dann posten!
- Suchfunktion und Betreffsuche benutzen - vielleicht gibt es schon einen ähnlichen Beitrag
- Aussagekräftigen Betreff wählen
- Im Betreff angeben um welchen Controllertyp es geht (AVR, PIC, ...)
- Groß- und Kleinschreibung verwenden
- Längeren Sourcecode nicht im Text einfügen, sondern als Dateianhang
- JPEG-Dateien (.jpg) nur für Fotos und Scans verwenden
- Schaltpläne, Screenshots usw. als PNG oder GIF anhängen
Formatierung (mehr Informationen...)
- [c]C-Code[/c]
- [avrasm]AVR-Assembler-Code[/avrasm]
- [pre]vorformatierter Text (z.B. Code in anderen Sprachen)[/pre]
- [math]Formel in LaTeX-Syntax[/math]
- [[Titel]] - Link zu Artikel


