Aus RN-Wissen.de
Version vom 17. Februar 2006, 10:43 Uhr von SprinterSB_alt (Diskussion | Beiträge) (Gültigkeitsbereiche)

Wechseln zu: Navigation, Suche
Rasenmaehroboter Test

Ein kurzer Einblick in die Programmiersprache C

Allgemeines

C wurde 1971 als Gundlage für das Betriebssystem UNIX in den USA entwickelt (UNIX ist zu über 90% in C geschrieben). 1978 wurde von Brian Kernighan und Dennis Ritchie eine eindeutige Sprachedefinition entwickelt. C ist mittlerweile von ANSI und ISO standardisiert. Heutzutage ist C (und ihr Nachfolger C++) die dominierende Programmiersprache. Sehr viele Anwendungen sind in C geschrieben. Leider ist C jedoch nicht einfach zu lernen, daher eignet es sich nur bedingt für Anfänger. Mit etwas Übung kann man damit jedoch sehr effiziente Programme schreiben.

C ist sehr eng an Assembler angelehnt, aber trotzdem Hardware-unabhängig. Das bedeutet, Sie können maschinennahe Programme sehr leicht (aber nicht ganz ohne Aufwand) auf ein anderes System portieren. Sie benötigen dazu lediglich einen anderen Compiler, und Inline-Assembler Anweisungen (Assembleranweisungen innerhalb eines C-Programmes) müssen der neuen Hardware (Prozessor) angepasst werden.

Geschichte

1971
C wird entwickelt
1978
Kernighan und Ritchie definieren die Sprache.
1983
ANSI und ISO standardisieren C.
1992
Bjarne Stroustrup enwickelt die Nachfolgesprache C++.

Aufbau eines C-Programmes

Ein einfaches C-Programm könnte folgendermassen aussehen. Das Programm tut eigentlich nichts, aber das Beispiel zeigt den prinzipiellen Aufbau.

#include <stdio.h>

int Zahl1;
char Zeichen1;

int main (void)
{
   int Zahl2;

   /* Anweisungen */
   return 0;
}

Die Beschreibung (die Erklärungen folgen):

#include <...>
Die Include-Direktive sagt dem Compiler, welche Header-Dateien er einbinden soll. In den Header-Dateien und den dazugehörigen Bibliotheken stehen Funktionen und Datentypen, die nicht im Compiler selbst implementiert sind, etwa komplexe Ausgabefunktionen wie "printf", die weiter unten erklärt wird. Durch den Include kann man solche Funktionen nutzen. Elementare Dinge hingegen, wie die mathematischen Operatoren +,-,*, etc. sind im Compiler selbst eingebaut.
int Zahl1;
Diese Anweisung definiert eine Variable vom Typ int (integer, eine ganze Zahl zwischen −32768 und +32767). Diese Variable ist im ganzen Programm gültig (global). Jede Anweisung in C wird mit einem Strichpunkt (Semikolon ;) abgeschlossen und dadurch von der nächsten Anweisung getrennt.
char Zeichen1;
Hier geschieht das selbe, nur wird diesmal eine Variable des Types char (character, ASCII-Zeichen oder Zahl zwischen -127 und +128) definiert.
int main (void)
definiert ein Unterprogramm mit dem Namen main, das keine Parameter hat (void) und eine ganze Zahl (int) zurückliefert. "main" ist das Hauptprogramm in C, wo mit der Ausführung nach dem Programmstart begonnen wird.
{
Die linke geschwungenen Klammer beginnt den Rumpf (auch "body" genannt) der main-Funktion. Danach folgen Variablendefinitionen, Kommentare und Anweisungen von main.
int Zahl2;
Innerhalb von "main" wird die lokale Variable Zahl2 definiert.
/* Anweisungen */
Das ist ein Kommentar in C. Hier kann man Anmerkungen zum Code hinschreiben oder Codestücke "auskommentieren", um sie zu deaktivieren. Der Kommentar beginnt mit /* und wird beendet mit einem /*. Er kann mehrere Zeilen überspannen. Je nach C-Compiler werden auch einzeilige Kommentare mit // akzeptiert, die nur bis zum nächsten Zeilenende reichen. Sie gehören jedoch nicht zum standard ANSI-C. Die Leerzeile nach dem Kommentar wird nicht weiter berücksichtige, sie kann zur Untergliederung des Codes zur besseren Lesbarkeit eingefügt werden.
return 0;
Gibt den Wert 0 zurück und beendet das Programm. Vor dem return können natürlich noch C-Anweisungen stehen, die aber erst weiter unten erklärt werden.
}
Die schliessende geschwungenen Klammer beendet den Rumpf des Hauptprogramms.

Das Unterprogramm main

Unterprogramme lernen Sie zwar erst später kennen, aber dieses eine muß sein. In Gegensatz zu anderen Programmiersprachen gibt es in C kein "Hauptprogramm", sondern nur Unterprogramme. Aber woher soll der Computer wissen, welches Unterprogramm er am Beginn ausführen soll? Ganz einfach, er führt immer das Unterprogramm "main" aus. Dieses ist sozusagen das "Hauptprogramm".

Der main-Funktion können beim PC Parameter übergeben werden. Dies sind die sogenannten Kommandozeilenparameter, die beim Aufruf eines Programmes hinter dem Dateinamen stehen. Zudem wird auch ein Integer Wert als Ergebnis zurückgeliefert. Beim Microcontroller ist main das Startprogramm, das nach dem RESET aufgerufen wird. Hier gibt es dann also keine Funktionsparameter. Ein Rückgabewert ist auch nicht sinnvoll, so daß main oft als oft auch als void (Parameterfrei) definiert wird, wie oben im Beispiel zu erkennen. Um Compilerfehler/Warnungen zu vermeiden, muss der Compiler dann aber mit speziellen Einstellungen gestartet werden, denn C-Standard ist, daß main einen Wert zurückliefert.

/* void Definition von main ist nur beim Controller üblich
 * spezielle Compilereinstellungen sind nötig, damit bei dieser Definition von main
 * kein Fehler/Warnung erzeugt wird. 
 */
void main(void)
{
}

Was das bedeutet, lernen Sie in einem späteren Kapitel. Wichtig ist nur: ohne main ist kein Programm komplett, es lässt sich dann auch nicht ausführen!

Blöcke

Im vorigen Abschnitt haben Sie bereits die geschwungenen Klammern { und } kennen gelernt. Doch was bedeuten Sie? Einem Pascal-Kenner ist das schnell erklärt: { entspricht BEGIN, } entspricht END. Wenn ihnen auch das unbekannt ist, dann hilft Ihnen hoffentlich die folgende Erklärung. Programme sind in Abschnitte unterteilt. Da gibt es zum einen das Hauptprogramm und die jeweiligen Unterprogramme, aber auch Schleifen und bedingte Anweisungen. Jedes dieser Beispiele stellt ein eigenständiges Stück Code dar. Daher müssen Sie es auch als solches kennzeichnen. Dies geschieht mit { und }. { bedeutet so viel wie "Block Anfang" und } bedeutet "Block Ende":

void main(void)
{    //der Block "main" beginnt
 int Zahl;
 
 {   //ein "namenloser" Block beginnt
   //hier stehen Anweisungen
 }   //der "namenlose" Block endet
}    //der Block "main" endet

Bemerkung: Der "namenlose" Block ist eigentlich unnötig. In den folgenden Kapiteln lernen Sie jedoch bessere Beispiele kennen!


Datentypen

Der Datentyp einer Variable gibt an, welche Werte eine Variable enthalten kann, und welcher Art diese Daten sind. So ist es zum Beispiel möglich, in eine Variable vom Typ int Zahlen zwischen ca. -32000 und +32000 einzutragen. In einer char-Variable können Sie alle möglichen ASCII-Zeichen speichern (alles, was Sie mit der Tastatur erzeugen können).

ACHTUNG: Da C plattformabhängig ist, hängt die Größe eines Datentypes zum Teil von der genutzten Hardware (z.B. 8, 16 oder 32 Bit Controller) und dem Compiler ab!

int, char, short, long (ganze Zahlen)

In Variable dieses Types können Sie ganze Zahlen abspeichern, also z.B. 1, -2, 100, 12345.

Größe (Bit) Typ Vorzeichen Grenzen des Wertebereichs
8 char signed
unsigned
−128
0
127
255
16 short signed
unsigned
−32.768
0
32.767
65.535
32 long signed
unsigned
−2.147.483.648
0
2.147.483.647
4.294.967.295
64 long long signed
unsigned
−9.223.372.036.854.775.808
0
9.223.372.036.854.775.807
18.446.744.073.709.551.615
8, 16, 32, 64
int signed
unsigned
plattformabhängig plattformabhängig

float, double Gleitkommazahlen

In einer Gleitkomma-Variable können Kommazahlen gespeichert werden, z.B. 3.141592654. Float reicht reicht für die meisten Kommazahlen. Werden jedoch noch höhere Genauigkeiten benötigt, kommt der Datentyp double zum Einsatz. Vorsicht: bei PIC (microchip) ist die innere Darstellung dieser Zahlen anders als bei meisten anderen Compilern, beim binären senden z.B. zum PC muß dann konvertiert werden !

char (Zeichen)

In einer vorzeichenbehafteten char-Variable können Sie Zahlen zwischen -128 und +127 abspeichern oder 0 bis 255 ohne Vorzeichen. Dieser Datentyp wird oft für ASCII-Zeichen genutzt .Für den Computer ist es ja egal, ob sich eine Zahl oder ein Zeichen in der Variable "befindet", er speichert alles in Form von binären Ziffernfolgen.

Dabei darf man eines nicht vergessen: Es macht einen großen Unterschied, ob man in einer char Variablen das Zeichen '1' (ASCII-Zeichen Nr. 49) abspeichert, oder die Zahl 1 (entspricht Hardware-technisch ASCII-Zeichen Nr. 1). Man kann zwar mit beiden rechnen, aber '1' * 2 ergibt nicht '2', sondern 'b' (ASCII-Zeichen Nr. 98)!

Strings (Zeichenketten)

Bei Strings handelt es sich um Wörter und Sätze, die aus einzelnen Zeichen gebildet werden. Die Ausgabe auf dem Bildschirm funktioniert am einfachsten mittels Strings. Wie dieser Datentyp genau funktioniert, erfahren Sie im Kapitel "besondere Datentypen".

Boolean (Logische Variablen)

In der Sprache C gibt es keinen Datentyp für boolsche Werte "wahr" bzw. "TRUE" oder "falsch" bzw. "FALSE". Statt dessen werden wird der Datentyp int dafür verwendet. Hat die jeweilige Variable den Wert 0, so ist sie FALSE, sonst (ungleich 0) ist sie TRUE. HINWEIS: In Zukunft werden ich oft 1 als Synonym für "ungleich 0" verwenden. Bitte beachten Sie jedoch, das eine Variable, die TRUE ist, nicht unbedingt den Wert 1 haben muß. Sie muß lediglich ungleich 0 sein!

Variablen

Eine Variable ist ein Synonym (=anderer Name) für eine Speicherstelle in einem Computer. Einfacher gesagt, eine Variable bietet Raum, um Daten (z.B. eine Zahl) zu speichern und wieder zu lesen. Dieser "Raum" wird im Schreib- und Lesespeicher (RAM) des Computers angelegt. Konstanten werden von manchen C-Compilern auch im Flash angelegt, da sie nicht veränderbar sind.

Anlegen von Variablen

Um eine Variable verwenden zu können, müssen Sie sie zuerst vereinbaren ("erzeugen"). Dies wird auch als "Definition der Variablen" bezeichnet und geht so: Schreiben Sie zuerst den Datentyp, dann den Namen der Variablen. Zum Schluß kommt noch der Strichpunkt. Und nicht vergessen: C unterscheidet zwischen Groß- und Kleinschreibung!

int Zahl1, Zahl2;
char Zeichen;

void main(void)
{
  float gleitZahl1;
  // Anweisungen
}

Erklärung: In einer Zeile können auch mehrere Variablen gleichen Types vereinbart werden, wenn man ein Komma dazwischen setzt. Variablen können in jedem "Block" vereinbart werden. Siehe Gültigkeitsbereiche.

Zuweisungen

Man kann einer vereinbarten Variable nun Werte zuweisen. Dazu schreibt man zuerst den Variablennamen, ein Gleichheitszeichen "=" und anschliessend den zuzuweisenden Ausdruck.

void main(void)
{
  int Zahl1, Zahl2 = 12;
  char Zeichen1 = 'A'.

  Zahl1 = 52;
  Zeichen1 = Zeichen1 + 1; 
}

Zuerst werden 3 Variablen angelegt (Zahl1, Zahl2, Zeichen1).

Zahl2
wird gleich bei der Vereinbarung der Wert 12 zugewiesen.
Zahl1=52
Hier wird der Variablen Zahl1 der Wert 52 zugewiesen.
Zeichen1
wird um 1 erhöht. Da in der Variablen 'A' gespeichert ist, gibt sich ihr neuer Wert aus 'A' + 1. Weil 'A' dem Wert 65 entspricht, ist 'A' + 1 gleich 66, was dem Wert für 'B' entspricht.

Zuweisungen bei float

Das funktioniert genau wie normale Zuweisungen. Nachkommastellen werden durch einen Punkt abgegrenzt:

floatVariable = 3.14;

Zusätzlich kann eine Zehlnerpotenz angegeben werden:

floatVariable2 = 1.234E-6;

Dadurch wird der erst Wert mit 10-6 multipliziert, der Wert der Variablen ist also

[math]1{,}234\cdot10^{-6} = 0.000001234[/math].

Zuweisungen bei logischen Variablen

Wie bereits erwähnt, besitzt C keinen logischen Datentyp. Es müssen also int und char dafür genutzt werden. Die Zuweisung entpricht der Standard-Zuweisung. Wird der Wert 0 zugewiesen, dann ist die Variable FALSE, ansonsten ist Sie TRUE.

intVariable = !0;   // entspricht TRUE
intVariable = 0;    // entspricht FALSE


Vergleich von Variablen

Sie können Variablen miteinander vergleichen. Das geschieht mit einem der folgenden Zeichen mit den normalen mathematischen Regeln:

Operator Bedeutung
== ist gleich
!= ist nicht gleich
< ist kleiner
<= ist kleiner oder gleich
> ist größer
>= ist größer oder gleich

Als Ergebnis bekommen Sie einen logischen Ausdruck in Form einer int-Zahl.

intVariable = Zahl1 <= Zahl2;

das komplette Beispiel zeigt es deutlicher:

 int i;
 int z1, z2;

 z1 = 5
 z2 = 100
 i = z1 <= z1  //Ein Vergleich. Die Integervariable i wird wahr, da z1 kleiner als z2
 printf ("Ergebnis: %d\n", i);

Die Variable i ist ungleich 0 (TRUE), wenn z1 kleiner oder gleich z2 ist. Ist z1 jedoch größer als z2, dann ist i gleich 0 (FALSE).

Konstanten

Konstanten können als Variable angesehen werden, die nicht beschrieben, sondern nur gelesen werden können. Ein typisches Beispiel dafür ist die Zahl [math]\pi[/math] (rund 3,141592654). Niemand würde in der realen Welt versuchen, ihr einen anderen Wert zuzuweisen. Würde man [math]\pi[/math] jedoch wie eine normale Variable anlegen, wäre dies ohne weiteres möglich. Um dies zu verhindern, gibt es dafür ein Schlüsselwort in C:

const datentyp name = value;  //Zuweisung bei der Defininition der Variablen

Wichtig dabei ist, dass man Konstanten nur bei der Vereinbarung einen Wert zuweisen kann. Da Konstanten gewöhnlich im gesamten Programm, zumindest einer Quelldatei genutzt werden, definiert man diese allerdings gewöhnlich außerhalb des main-Blockes entweder am Anfang eines Programmes, oder in einer sogenannten Header-Datei, die per include eingebunden wird.

const float pi = 3.141592;  //Zuweisung bei der Defininition der Variablen

void main(void)
{
  //Hauptprogram
}

Gültigkeitsbereiche

In C können mehrere Variablen den gleichen Namen haben, solange eindeutig ist, welche in welchen Block gültig ist. Dabei gelten folgende Regeln:

Lokale Variablen
sind Variablen, die innerhalb eines Blockes definiert werden. Jede Variable ist nur in dem Block gültig, in dem Sie vereinbart wurde, sowie in allen darin enthaltenen Blöcken, es sei denn, in einem Unter-Block wird eine Variable gleichen Namens definiert. Dann bezieht sich in diesem Unter-Block der Bezeichner auf die im Unter-Block angelegte Variable.
Globale Variablen
werden ausserhalb jedes Blockes definiert und gelten ab der Stelle, an der sie deklariert werden. Wird jedoch in einem Block eine Variable gleichen Namens angelegt, gilt ab hier bis zum Ende des Blocks nicht mehr die globale Variable, sondern die im Block definierte. Das Spiel kann man weiterspielen: wird in einem Unter-Block wieder eine namensgleiche Variable angelegt, gilt diese in dem Unterblock.

Speicherklassen

Jede Variable in C gehört zu einer bestimmten Speicherklasse

auto
Lokale Variablen sind in aller Regal sogenannte automatische Variablen. Das bedeutet, sie werden automatisch angelegt, wenn ein Block bzw. eine Funktion betreten wird und danach wieder entfernt. Das Schlüsselwort "auto" wird praktisch nie hingeschrieben, denn lokale Variablen ohne die ausdrückliche Angabe einer Speicherklasse, sind automatisch automatische Variablen.
register
Durch diese Speicherklasse wird eine Variable – falls möglich – als Registervariable angelegt, also in einem Maschinenregister der Computer/Controllers gehalten. Dadurch kann auf solche Variablen besonders schnell zugegriffen werden.
extern
Baustelle
static
Baustelle

Ausdrücke

Eine Variable oder eine Konstante in C stellen einfache Ausdrücke dar. Diese elementaren Ausdrücke können durch Operatoren miteinander verknüpft werden und so zu neuen, komplexeren Ausdrücken zusammen gesetzt werden.

Einfache Beispiele für Ausdrücke sind also z.B.:

1
a
'a'
1 + a
a == 1

Auch Funktionen können einen Wert zurückliefern und in Ausdrücken weiter benutzt werden. In den folgenden Abschnitten wird gezeigt, welche Operatoren in C vorhanden sind, und wei man damit neue Ausdrücke aufbauen kann.

Lvalues

Ein LValue in C ist ein Ausdruck, dem ein anderer Ausdruck zugewiesen werden kann. das 'L' leitet sich ab von 'left' bwz. 'links' und das 'value' bedeutet Wert. Eine Variable ist so ein LValue, der etwas zugewiesen werden kann:

a = 1;

Hingegen ist a+1 kein LValue, denn eine Zuweisung wie

a+1 = 2;

erzeugt einen Compilerfehler, der etwas lautet könnte "illegal lvalue in assignment": "ungültiger Wert in Zuweisung"

Logische (boolsche) Operatoren

Ausdruck Beschreibung
a && b wahr, wenn a wahr und b wahr
a || b wahr, wenn a wahr oder b wahr
a == b gleich
a != b ungleich
a <= b kleiner oder gleich
a < b kleiner als
a >= b glösser oder gleich
a > b grösser als
!a wahr, wenn a nicht wahr und vice versa

Arithmetische Operatoren

Ausdruck Beschreibung
a + b Summe (Addition)
a - b Differenz (Subtraktion)
a * b Produkt (Multiplikation)
a / b Quotient (Division, evtl. mit Rest)
a % b Rest bei Division (Modulo)
-a Vorzeichenumkehr (Zweierkomplement)

Bit-Operatoren

Ausdruck Beschreibung
a & b bitweise und (and)
a | b bitweise oder (or)
a ^ b bitweise exclusiv-oder (xor, exor)
~a jedes Bit in a invertieren (not, Einerkomplement)

Index-Operator bei Arrays

Komponenten-Auswahl bei Structs und Unions

Adress-Operator und Dereferenzierung

Cast-Operator

Der Cast Operator dient dazu, den Datentyp eines Wertes zu ändern. Dafür wird einfach der neue Datentyp in Klammern vor den Wert geschrieben.

Um zum Beispiel aus einem Float ein Integer zu machen:

var  = (int) 5.60;

Dabei wird der Wert aber auch gerundet, und es findet somit ein Informationsverlust statt.

Ein weiteres Beispiel ist das Umwandeln einer ganzen Zahl in eine Adresse:

int * addr;
addr = (int*) 0x1234;

Damit ist addr ein Zeiger auf einen int an Adresse 0x1234.

Komma Operator

Mit einem , können mehrere Ausdrücke nacheinander ausgewertet werden. Die Auswertung erfolgt von links nach rechts.

Solche Konstrukte sieht man manchmal in Abfragen wie

FILE  *file;
if (file = fopen ("foo.exe", "r"), file != NULL)

was erst an file einen Wert zuweist und den if-Block nur betritt, wenn file nicht der Nullpointer ist.

Zuweisungen und Operatoren mit Nebeneffekt

Bedingte Zuweisung

lvalue = (bediungung) ? if-ausdruck : else-ausdruck;

Blöcke

Selbst Blöcke haben in C einen Wert, der in einer Berechnung verwendet werden kann. Der Wert eines Blockes ist der Wert der letzten Anweisung des Blocks. Der Block muss un runden Klammern stehen:

int sum;
sum = ({
   int i, s = 0;
   for (i=1; i<= 100; i++)
      s += i;
   s;
   });

Ein weiteres Beispiel ist das Makro PSTR von avr-gcc, das einen String ins Flash legt und die Adresse zurückliefert, was nur in einer Instruktion nicht machbar wäre:

#define PSTR(s)                         \
  ({                                    \ 
      static char __c[] PROGMEM = (s);  \
      __c;                              \
  })

Reihenfolge der Auswertung

Kontrollanweisungen

Boolsche Logik

Boolean-Variablen können miteinander verknüpft werden. Dies geschieht mit den boolschen Operatoren && ( logisches "und"), || (logisches "oder") und ! (logisches "nicht") (es gibt noch weitere, auf die hier nicht eingegangen wird).

b1= b2 && b3;


Der Wert von b1 ist dann 1 (TRUE), wenn b2 und ("AND") b3 1 (TRUE) sind. Ist eine der beiden (oder auch beide) 0 (FALSE), dann ist b1 ebenfalls 0 (FALSE).


b1= b2 || b3;


Der Wert von b1 ist dann TRUE, wenn b2 oder ("OR") b3 TRUE ist. Ist also einer (oder beide) TRUE, dann ist auch b1 TRUE, sind beide FALSE, ist auch b1 FALSE.


b1= !b2;


Der Wert von b1 ist dann TRUE, wenn der von b2 FALSE ist. Die NOT Verknüpfung "dreht den logischen Wert der Variablen um".

IF - Anweisung

Syntax: IF (Bedingung) Anweisung; Mit Hilfe der IF Anweisung kann man Codeteile (Blöcke) ausführen lassen, wenn die dazugehörige Bedingung erfüllt (TRUE) ist. Im Flussdiagramm sieht eine If-Anweisung folgendermassen aus:

Ifthenelse.png

Und in C schreibt man:

if (Bedingung)        // WENN
{                     // DANN
       Anweisung1;
}
else
{                     // SONST
       Anweisung2;
}

Wenn die Bedingung wahr ist wird Anweisung1 ausgeführt. Anschliessend wird aus der If-Bedingung herausgesprungen. Anweisung2 wird also nicht ausgeführt.

Ist die Bedingung unwahr wird gleich zu Anweisung2 gesprungen, diese ausgeführt, und anschliessend wird die If-Bedingung beendet.

Achtung!

Ein häufiger Fehler ist statt if(a == 23) etwas wie if(a = 23) zu schreiben. Hier wird allerdings nicht geprüft ob die Variable a gleich 23 ist, sondern der Variable a wird der Wert 23 zugewiesen, und das ist eine wahre Aussage. Anschliessend wird demzufolge der "DANN"-Teil ausgeführt.

Die Syntax hierbei ist allerdings korrekt, der Compiler wird also keinen Fehler ausspucken. Somit ist dieser kleine Fehler sehr schwer zu finden. Abhilfe schafft die Schreibweise if(23 == a). Wenn man dort anstatt des Vergleichsoperator '==' den Zuweisungsoperator '=' verwendet spuckt der Compiler sehr wohl einen Fehler aus.


Ein weiterer häufiger Fehler ist das schreiben von if(Bedingung); Richtig müsste es heissen "if(Bedingung)" Das fehlerhafte Semikolon im ersten Fall interpretiert der Compiler bereits als Anweisung. Auch hier liegt kein Syntaxfehler vor und der Compiler schweigt. Hier also ebenfalls besser zweimal hinschauen.

Schleifen

Um Anweisungen mehrmals hintereinander auszuführen, benötigt man Schleifen. Diese führen Anweisungen aus, bis oder solange Bedingungen erfüllt sind.
Wichtig ist also, ob die Bedingung vor oder nach den Schleifen-Anweisungen geprüft wird.

WHILE-Schleife

Syntax: while (Bedingung) { Anweisung; }

Die WHILE-Schleife wird solange durchlaufen, wie die Bedingung erfüllt (TRUE) ist. Die Schleife wird also unter Umständen garnicht durchlaufen. Bei mehreren Anweisungen benötigen man einen Block!

Zahl1=0;
while (Zahl1<3)
{
   Zahl1=Zahl1+1;
   Zahl2=Zahl2*2;
}

In diesem Beispiel wird die Schleife 3 mal durchlaufen. Zu Beginn des 4. Durchlaufes ist die Bedingung FALSE (Zahl1 ist dann nicht mehr kleiner, sondern gleich 3!), also wird mit dem Befehl nach der Schleife fortgesetzt.

DO - WHILE-Schleife

Syntax: do { Anweisung; }while (Bedingung);

Die DO WHILE-Schleife wird auf jeden Fall einmal durchlaufen und dann solange wiederholt, wie die Bedingung erfüllt ist. Das ist dann sinnvoll, wenn der Vergleichswert für die Bedingung erst im Anweisungsblock gesetzt wird

unsigned char bData;
do {
    bData = getch();   // Tastatureingabe (s.u.) 
}while (bData != 13);

Die Schleife wird durchlaufen und wiederholt, solange etwas anderes als 13 (<ENTER>) eingegeben wird.

FOR-SCHLEIFE

Syntax: for (Zuweisung;Bedingung;Inkrement) { Anweisung; }

Die FOR-Schleife ist eine spezielle WHILE-Variante für die Formulierung von Zählschleifen. Die angegebene Variable wird am Anfang auf einen Wert gesetzt. Danach wird sie nach jedem Durchlauf verändert (meist um 1 erhöht). Hat sie einen bestimmten Wert erreicht, wird die Schleife beendet. Innerhalb des Anweisungsblockes können Sie auf die Variable zugreifen, d.h. Sie können sie auch verändern. Das ist jedoch nicht ratsam. Wollen Sie mehrere Anweisungen ausführen, benötigen Sie eine Block

Die gesammte Schleife wird solange durchgeführt, wie die Bedingung TRUE ist (d.h. Sie brauchen garnicht die Zählvariable abfragen, Sie können hier auch jede andere Variable auswerten). Inkrement letztendlich wird nach jedem Schleifendurchlauf ausgeführt. Hier kann man z.B. den Zähler erhöhen.

Die Theorie ist schwer, die Praxis nicht so sehr...

int Zaehler;
for (Zaehler = 1; Zaehler <= 10; Zaehler++) 
{
   Zahl2 = Zahl2 + Zaehler;
}

Das Äquivalent als "WHILE"

int Zaehler;
     Zaehler = 1;                      // Anfangswerte
     while (Zaehler <= 10)             // Bedingung
     {
        Zahl2 = Zahl2 + Zaehler;
        Zaehler++;                    // Inkrement
     }


In diesem Beispiel wird Zahl2 in jedem Schleifendurchlauf um die Zählvariable Zaehler erhöht. Da Zaehler nacheinander die Werte von 1 bis 10 hat, ist in Zahl2 nach der Schleife die Summe der Zahlen 1 bis 10 gespeichert.

Erklärung: Zaehler=1 bedeutet, dass der Variablen Zaehler vor dem 1. Schleifendurchlauf der Wert 1 zugewiesen wird. Zaehler<=10 ist die Schleifenbedingung; ist sie FALSE, wird die Schleife beendet. Zaehler++ bedeutet, dass Zaehler nach jedem Durchlauf um 1 erhöht wird.

Ein- und Ausgabe

Bildschirm-Ausgabe

Bisher war das Tutorial trotz aller Beispiele reine Theorie. Sie konnten zwar Programme schreiben, aber die Funktion nicht testen. Hier lernen Sie nun, wie Sie etwas am Bildschirm ausgeben.

Der dazu notwendige Befehl ist printf (das f ist KEIN Fehler!). Diese Anweisung gibt die ihr übergebenen Parameter auf dem Bildschirm aus (außer, Sie haben unter DOS die Standard-Ausgabe auf ein anderes Gerät, z.B. den Drucker umgeleitet, aber das ist eine andere Geschichte). Sie kann beliebig viele Parameter übernehmen. Es müssen jedoch Standard-Datentypen (z.B. int, char, float, double) sein!

#include <conio.h>
#include <stdio.h>

void main(void)
{
    int Zahl1 = 12;
    char Zeichen1 = 'A';
    
    printf ("Das ist Text, und er wird als solcher ausgegeben. \n");
    printf ("Der Wert der Variablen Zahl1 ist: %d \n", Zahl1);
    printf ("Der Wert der Variablen Zeichen1 ist: %c \n", Zeichen1);
    printf ("Der Wert der Variablen Zeichen1 ist: %d \n", Zeichen1);
}

Hinweis: Dass das Programm nach dem Compilieren und Ausführen sofort wieder beendet wird, ist nicht Ihr Fehler. Drücken Sie [ALT] + [F5], um zum Ausgabebildschirm zu kommen! Erklärung:

Der 1. printf Befehl gibt Text aus. Das Zeichen am Ende (\n) bedeutet "New Line", es bewegt den Cursor an den Anfang der nächsten Zeile.

Der 2. printf-Befehl gibt auch Text aus, am Ende befindet sich wieder das \n, um einen Zeilenvorschub zu erreichen. Das %d wird vom Compiler durch den 1. Parameter ersetzt, der nach dem Text angegeben wird. In diesem Fall wird %d also durch den Wert der Variablen Zahl1 ersetzt. Das d im %d bedeutet "Dezimalzahl", der Computer gibt also eine ganze Zahl aus.

In der 3. Ausgabe wird ein Zeichen ausgegeben. Diesmal bedeutet %c "char" (Zeichen). Es wird also %c durch 'A' ersetzt.

Die letzte Ausgabe verhält sich merkwürdig – es wird doch tatsächlich eine Zahl ausgegeben, obwohl der Parameter ein Zeichen (char) ist. Aber: %d im Text bedeutet, dass eine Zahl auszugeben ist. ein char kann ja auch als Zahl interpretiert werden. Es wird also 65 (der ASCII-Wert von 'A') ausgegeben. Das ist ein typisches Beispiel für das mögliche unterschiedliche Interpretieren einer char Variablen!

Bildschirm löschen

Oft sind von anderen Programmen noch "Rückstände" vorhanden. Nicht nur um die zu beseitigen, brauchen Sie einen "Bildschirm-Lösch-Befehl". Dieser ist in conio.h definiert und nennt sich "clrscr" ("clear screen", deut: "lösche Bildschirm").

#include <conio.h>
#include <stdio.h>

void main(void)
{
  clrscr();
  ...
}

Tastatur-Eingabe

Um ein "gscheites" Programm schreiben zu können, muß man wissen, wie der Benutzer über die Tastatur Befehle eingeben kann. Die dafür notwendigen Funktionen stelle ich in diesem Kapitel vor. Der wichtigste Befehl ist "scanf". Er liest Daten von der Tastatur. Die Syntax entspricht derer von printf, nur daß man außer den %c und %d keine weiteren Angaben im "Text" machen darf:

printf("Bitte geben Sie eine Zahl ein: ");
scanf("%d",&Zahl1);
printf("Geben Sie einen Zeichen ein: ");
scanf("%c",&Zeichen1);

Das Programm gibt eine Eingabeaufforderung aus. Dann erwartet es vom Benutzer, daß er eine Zahl eingibt, die mit [ENTER] bestätigt wird. Diese Zahl wird in Zahl1 abgespeichert. Danach erfolgt wiederum eine Aufforderung zur Eingabe, diesmal eines einzelnen Zeichens. Dieses kann man nun eingeben und ebenfalls mit [ENTER] bestätigen.

Macht man keine dem Datentyp der erwarteten Variable entsprechende Eingabe, dann bricht das Programm mit einer Fehlermeldung ab (wenn man z.B. "1_T2" eingibt, wenn eine Zahl erwartet wird)!

Das & vor den Parametern ist notwendig. Warum, das erfahren Sie im Kapitel "Unterprogramme". Für die Profis eine Kurz-Erklärung: Das Unterprogramm scanf bekommt zwar einen Wert übergeben, kann aber keinen zurückliefern ("call by value"). Daher wird kein Wert, sondern ein Zeiger auf eine Variable übergeben. Mit dem & Zeichen bekommen Sie die Adresse einer Variablen ("call by reference").

getch

getch ist eine Funktion (Siehe Kapitel Unterprogramme), die ein char-Zeichen von der Tastatur einliest. Dazu geben Sie im Programm einfach folgende Zeile an:

  char zeichen;
  zeichen = getch();

Diese Zeile erwartet die Eingabe eines char-Zeichens. Im Gegensatz zu scanf muß man nach der Eingabe des Zeichens jedoch nicht [ENTER] drücken, um die Eingabe zu bestätigen. In der Variable zeichen wird das erste Zeichen gespeichert, das auf der Tastatur eingegeben wird.

Eine weiter Möglichkeit der Anwendung von getch ist es, wenn Sie einfach "getch();" als Befehl eingeben. Dann wartet der Computer, bis Sie eine Taste drücken, speichert diese aber nicht ab.

kbhit

Dies ist ebenfalls eine Funktion, die eine boolschen Ausdruck zurück liefert. Sie gibt Bescheid darüber, ob eine Taste gedrückt wurde, und mit scanf oder getch abgerufen werden kann (zum Verständnis: auf der Tastatur eingegebene Zeichen werden meist nicht direkt vom Programm verarbeitet, sondern erst vom Betriebssystem bzw. dem BIOS des Computers gespeichert. Wenn das Programm nun eine Eingabe erwartet, werden die gespeicherten Zeichen "verwendet". Dies lauft jedoch so schnell ab, daß der Benutzer davon nichts merkt). Ist diese Variable TRUE, dann ist ein Zeichen abfragebereit, sonst nicht:

while ( !kbhit() )
{
  Zahl1=Zahl1+1;
  printf("%d\n",Zahl1);
}

Dieser Programmteil erhöht solange Zahl1 und gibt diese aus, bis der Benutzer eine Taste drückt.


Allgemeines zu Unterprogrammen

Stellen Sie sich vor, Sie haben einen eine Code-Folge, die mehrmals im Programm vorkommt, z.B. eine mathematische Formel. Anstatt dieses Codestück mehrmals zu schreiben (was Zeit beim Erstellen und Speicherplatz im ausführbaren Programm kostet), können Sie den Abschnitt in ein sogenanntes "Unterprogramm" schreiben. Dieses Unterprogramm können Sie dann von jeder Stelle ihre Hauptprogrammes aus aufrufen. Anders als in anderen Programmiersprachen gibt es in C keine generelle Unterscheidung zwischen Funktionen und Prozeduren. Trotzdem möchte ich kurz darauf eingehen:

Funktionen

Syntax: Rückgabe-Typ Name(Parameterliste);

Funktionen sind Unterprogramme, die am Ende der Ausführung einen Wert zurückliefern, wie z.B. getch. Paradebeispiele für die Anwendung einer Funktion sind jedoch mathematische Formeln:

#include <conio.h>
#include <stdio.h>

int hoch2(int param1)
{
  int zahl;
  zahl=Param1*Param1;
  return zahl;
}

void main(void)
{
  int zahl1,ergebnis;
  
  printf("Bitte Zahl eingeben: ");
  scanf("%d",&zahl1);
  ergebnis = hoch2(zahl1);
  printf("%d hoch 2 = %d\n",zahl1,ergebnis);
  printf("5 hoch 2 = %d\n",hoch2(5));
  getch();
}

Bemerkungen zum Programm: Ein Unterprogramm kann an jeder beliebigen Stelle innerhalb eines Programmes stehen, aber nur außerhalb von Blöcken. Siehe auch Prototypen. Geschachtelte-Unterprogramme wie z.B. in Pascal sind nicht möglich. In der Deklaration der Funktion kommt zuerst der Rückgabe-Datentyp. Jede Funktion liefert einen Wert zurück, dieser Datentyp gibt den Typ der Variablen an. Bei z.B. getch() ist das char. Danach folgt der Name der Funktion (case-sensitiv!). Danach kommt in Klammern die Parameter-Liste. Dabei handelt es sich um eine Liste von Parametern, die jeweils durch Typ und Name angegeben werden, und mittels Beistrich voneinander getrennt werden. Am Ende ist kein Beistrich! Im Block der Funktion können alle Anweisungen ausgeführt werden und Variablen deklariert werden. Am Ende der Funktion muß diese ordnungsgemäß beendet werden. Dies geschieht mittels der Anweisung "return" und dem Rückgabewert. Um eine Funktion aufzurufen (aus einem anderen Unterprogramm oder main), geben Sie folgendes an "Variable=Funktion(Parameter);". Bei Variable muß es sich jedoch nicht unbedingt um eine von Ihnen deklarierte Variable handeln. Es kann z.B. auch der Parameter für ein weiteres Unterprogramm sein (wie im Beispiel, letzte Zeile). Die Einrückungen der Anweisungen innerhalb der Funktion nach rechts ist nicht notwendig, sie dient jedoch der Übersicht und ist sehr empfehlenswert.

Prozedur

Syntax: void Name(Parameterliste);

Eine Prozedur ist nichts anderes als eine Funktion ohne Rückgabewert. Sie ist daher prinzipiell gleich aufgebaut wie eine Funktion, bis auf die folgenden Unterschiede:

Als Rückgabe-Datentyp wird void angegeben. Eine Prozedur kann daher beim Aufruf nicht einer Variablen zugewiesen werden (wie FUNCTION's) sonder wird wie eine normale Anweisung behandelt (wie z.B. printf).

#include <conio.h>
#include <stdio.h>

void ausgeben(int Param1);
{
  printf("Der Wert der Variablen beträgt: %d\n",Param1);
}

void main(void)
{
   int Zahl1;
   ausgeben(23);
   Zahl1=4;
   ausgeben(Zahl1);
   getch();
}

Bemerkungen zum Programm: Durch den Befehl "ausgeben(23)" wird die Prozedur "ausgeben" mit dem Parameter 23 aufgerufen. Param1 hat also den Wert 23. Die einzige Anweisung innerhalb der Prozedur ist printf, was nun ausgeführt wird. Danach wird mit dem nächsten Befehl innerhalb des Hauptprogrammes fortgefahren ("Zahl1=4"). Der 2. Aufruf von ausgeben erfolgt mit dem Parameter Zahl1, also wird Param1 der Wert 4 zugewiesen. Dieser wird nun wieder mittels printf ausgeben. Unterprogramme ohne Parameter Das ist ihnen sicherlich schon aufgefallen: Viele Unterprogramme, z.B. main, haben statt der Parameter immer nur void stehen. Was bedeutet das? Nun ja, C erwartet, das Sie die Parameter angeben. void bedeutet "es gibt keine Parameter". "void main(void)" bedeutet also: "Erstelle ein Unterprogramm Namens main, das keinen Rückgabewert und keine Parameter hat".

Merke: Auch wenn eine Funktion keine Parameter hat, z.B. getch, müssen Sie beim Aufruf trotzdem die Klammern angeben. Der Befehl "getch;" führt nicht getch ohne Parameter aus, sondern liefert die Adresse der Funktion getch. Alles klar? Na ja, jedenfalls: Beim Aufruf jedes Unterpgrogrammes immmer Klammern angeben.

Prototypen

Wie oben erwähnt, kann ein Unterprogramm an jeder beliebigen Stelle im Programm stehen. Damit ist jedoch eine Bedingung verknüpft: Das Unterprogramm muß in der Datei oberhalb des 1. Aufrufes definiert worden sein. Wenn Sie ein Unterprogramm in Zeile 10 zum ersten mal aufrufen, müssen Sie die Definition davor erledigt haben. Verstanden? Um dies zu erreichen, gibt es zwei Möglichkeiten:

Entweder Sie schreiben alle Unterprogramme vor main in die Datei. Dies muß jedoch wiederum so geschehen, dass Funktionen zum Zeitpunkt ihres Aufrufes bereits bekannt sind! Wo dies nicht möglich ist (z.B. sich gegenseitig aufrufende, rekursive Unterprogramme), oder wenn Sie das stört, müssen Sie Prototypen verwenden. Wie definiert man nun Prototypen? Sie kopieren einfach die erste Zeile des Unterprogrammes (z.B. void ausgeben(int Zahl); ) und fügen es an einer geeigneten Stelle ein (so, dass alle Aufrufe später in der Datei kommen). Beachten Sie jedoch, daß Prototypen einen Strichpunkt am Ende haben! Solche Definitionen stehen gewöhnlich am Anfang des der Quellcodedatei oder in einerm Header-Datei die eingebunden wird.

#include <conio.h>
#include <stdio.h>

void ausgeben(int Zahl);  //Der Prototyp

void main(void)
{
   clrscr();
   ausgeben(12);
   getch();
}

void ausgeben(int Zahl)   //Die eigentliche Prozedur
{
  printf("Ausgabe: %d\n",Zahl);
}

Besondere Datentypen

Felder

Oft muß man sehr viele Werte gleichzeitig abspeichern und betrachten, die alle der selben Aufgabe dienen. Man schreibt z.B. ein Programm, das 10 Zahlen einlesen und anschließend wieder ausgeben soll. Man könnte das natürlich mit 10 einzelnen Variablen bewerkstelligen, aber es ist sinnvoller, dabei Felder, sogenannte "ARRAY"s, zu verwenden.

In einem Array werden mehrere Variablen gleichen Typs zusammengefasst und hintereinander im Speicher abgelegt. So kann man viele tausend Variablen anlegen mit nur einer Zeile Code. Doch es gibt noch größere Vorteile: Sie können das Array mit z.B. einer for-Schleife ganz einfach nach Werten durchsuchen. Stellen Sie sich vor, Sie müssten mit 100 verschiedenen Variablen Zahl_001 bis Zahl_100 arbeiten! Ein ihnen bereits bekanntes Beispiel für ein Array ist ein String. Ein String ist nichts anderes als ein Array des Datentypes char.

Syntax: Datentyp Variablenname[Anzahl]

Der Name muß natürlich ein gültiger Bezeichner sein, als Datentyp kann jeder Typ genommen werden. In der eckigen Klammer wird die Anzahl der Elemente bekanntgegeben. Ein mit [3] definiertes Array hat Platz für 3 Variablen. Da der Index immer bei 0 beginnt, graft man also mit [0], [1] und [2] auf den jeweilige Inhalt zu. Um auf eine der im Array enthaltenen Variablen zugreifen zu können, müssen Sie den Variablennamen und in eckigen Klammern den Index (die "Nummer") der Variablen angeben. Diese Variable verhält sich dann wie eine ganz normale Variable des jeweiligen Datentypes.

#include <conio.h>
#include <stdio.h>

void main(void)
{
  int Zaeler;
  int Zahlen[10];  //Zahlen[0] bis Zahlen[9] !!!
  clrscr();
  for (Zaehler=0; Zaehler<10; Zaehler++)
  {
    printf("Bitte Zahl %d eingeben: ",Zaeler);
    scanf("%d",Zahlen[Zaehler]);
    printf("\n");
  }
  printf("Super!\n");
  for (Zaeler=0; Zaeler<10; Zaeler++) printf("Zahl %d war: %d\n",Zaehler,Zahlen[Zaehler]);

Bemerkungen: Zuerst wird ein 10 int-Variablen großes Array angelegt. In dieses wird nun der Reihe nach 10 Zahlen eingelesen. Anschließend werden alle 10 Zahlen ausgegeben.

Merke: Wenn Sie einen ungültigen Index angeben (einen, der in der Deklaration nicht enthalten war) können je nach Compiler und Einstellung undefinierbare Dinge passieren da dadurch andere Variableninhalte oder Programmcode überschrieben wird. Im schlimmsten Fall könnte sogar der Computer / Controller abstürzen. Also achten Sie darauf, daß Sie keine ungültigen Werte als Index angeben!

Mehrdimensionale Felder

Manchmal benötigen mehr als nur ein ein-dimensionales Array, wie Sie es bisher kennengelernt haben. Auch dies ist kein Problem. In der Deklaration geben Sie einfach mehrere eckige Klammernhintereinander an. Aber Vorsicht: der Speicherplatz ist begrenzt, ein "char feld[1024][1024]" hat die Speicherplatzgrenzen bereits weit überschritten, und der Compiler wird einen (bei gewissen Einstellung auch keinen) Fehler liefern. Beim Zuweisen von Werten etc. müssen Sie bei mehrdimensionalen Feldern auch mehr als einen Index angeben, die Erklärung folgt als Beispiel:

#include <conio.h>
#include <stdio.h>

void main(void)
{
  int x,y;
  int feld[3][5];
 
  for(x=0; x<3; x++) for (y=0; y<5; y++)
  {
    printf("Feldwert X: %d Y: %d",x,y);
    scanf("%d",feld[x][y]);
    printf("\n");
  }
  clrscr();
  for(x=0; x<3; x++) for (y=0; y<5; y++) printf("Wert: feld[%d][%d]= %d\n",x,y,feld[x][y]);
}

Erklärung:

Zuerst wird ein 3 mal 5 int Array angelegt. Dann werden die Werte eingegeben: zuerst feld[1][1], dann feld[1][2], usw. bis feld[3][5]. Zum Schluß werden alle Werte noch einmal ausgegeben.

Strings

Im Kapitel Variablen und Datentypen haben Sie bereits etwas von Strings gehört, mangels Wissen über Felder und Unterprogramme aber nicht sehr ausführlich. Hier folgt nun die ausführlichere Erklärung. Bei Strings handelt es sich um char Felder. Wenn Sie einen String definieren, müssen Sie das mittels char machen:

char string[21];

Nun haben Sie eine string, in dem Sie 21 Zeichen speichern können. Ganz richtig ist das jedoch nicht. C arbeitet mit "null-terminierten Strings". Das beudeutet, dass die Länge des Strings nicht abgespeichert wird (wie in Pascal), sondern ein Zeichen innerhalb des Strings das Ende desselben angibt (ASCII Nr. 0, daher der Name "null terminiert"). Das letzte Zeichen eines Strings muß daher immer das ASCII-Zeichen Nr. 0 sein. Ist es das nicht, hat der String kein definiertes Ende, und wenn Sie versuchen, ihn auszugeben, könnte es eine Weile dauern, bis sich im Speicher zufällig irgendwo eine 0 befindet. Es stehen ihnen daher (im Beispiel) nur 20 Zeichen zur Verfügung.

strcpy

Bei vielen Compilern können sie einem String nicht direkt einen Wert (Text) zuweisen. Dazu müssen Sie dann die Prozedur strcpy() benutzen. Diese erwartet als ersten Parameter den Namen einer String-Variablen (ohne eckige Klammern) und als zweiten Parameter den eines (anderen) Strings. Letzterer kann auch ein in doppelten Hochkommas (") eingeschlossener Text sein. Die Funktion fügt am Ende automatisch ein 0-Zeichen ein. Um diese Funktion nutzen zu können, müssen Sie die Datei string.h includieren!

#include <conio.h>
#include <stdio.h>
#include <string.h>

void main(void)
{
  char stri1[21],eingabe[21];

  strcpy(stri1,"hallo");
  clrscr();
  printf("Der 1. String: %s\n",stri1);
  printf("Bitte geben Sie maximal 20 Zeichen ein: ");
  scanf("%s",eingabe);
  strcpy(stri1,eingabe);
  printf("\n %s = %s",stri1,eingabe);
  getch();
}

Hinweis: Da ein String, wie jedes Feld, eigentlich ein Zeiger ist, brauchen Sie kein & bei scanf angeben!

Erklärung: Es werden 2 gleich große Strings definiert: stri1 und eingabe, mit je 20 "nutzbaren" Zeichen. stri1 wird der Wert "hallo" zugewiesen. Das 0-Zeichen wird automatisch angefügt! Der String wird ausgegeben. Als neues "Sonderzeichen" kommt %s ins Spiel. Es hat die gleiche Aufgabe wie %d oder %c, nur halt für Strings. Sie werden gebeten, eine String einzugeben. Dieser String wird danach in die Variable stri1 kopiert. Beide Strings (die ja nun auch den selben Wert haben) werden ausgegeben.

strlen

Die Funktion strlen, die als Parameter eine String-Variable erwartet, liefert die Länge diese Strings zurück. Sie werden jetzt vermutlich sagen: "DAs ist doch klar, wie lang der String ist. Ich habe es ja bei der Deklaratin angegeben". Das stimmt schon, aber denken Sie noch einmal an die null-terminierten Strings. Das 0-Zeichen steht am Ende des Strings (am Ende der gültigen Zeichenfolge), aber nicht unbedingt am Ende des reservierten Speicherplatzes. Haben Sie eine Variable "char Variable[21];", und ihr den Wert "hallo" zugewiesen, dann steht das null-Zeichen in Variable[5]. Der "gültige" String ist also 5 Zeichen (0-4) lang. Und genau das (5) würde strlen zurück liefern.

#include <conio.h>
#include <stdio.h>
#include <string.h>

void main(void)
{
  char stri[21];
  strcpy(stri,"hallo");
  printf("Der String ist %d Zeichen lang", strlen(stri) );
  getch();
}

Diese Funktion wird vor allem gebraucht, wenn Sie direkt auf den String zugreifen, mittels stri[0], stri[1], etc.


Strukturen

In C können Sie sogenannte "Strukturen" definieren. Dabei handelt es sich um eine Zusammenfassung von mehreren Datentypen zu einem größeren. Im Unterschied zu Feldern können in Strukturen jedoch unterschiedliche Datentypen gespeichert werden:

#include <conio.h>
#include <stdio.h>

struct PERSON {
  char vname[20],nname[20];
  char telnr[15];
  int alter;
};

Erklärung: "struct {" leitet die Definition ein. Dann werden 4 Variablen angelege: 3 Strings und ein int. mit } wird der Block wieder geschlossen, anschließend muß der Struktur ein Name zugewiesen werden (z.B. PERSON). Sie haben damit einen Datentyp erstellt. Um eine Variable dieses Typs (PERSON) anzulegen, geben Sie einfach "PERSON Variablenname;" an. Hinweis: Das der Datentyp groß geschrieben wird, ist nicht notwendig. Es dient, wie schon so oft, nur der Übersichtlichkeit. Wie spricht man die einzelnen Variablen einer Struktur nun an? Das funktioniert so:

Zuerst gibt man die Struktur-Variable an, im Beispiel "hubert" oder "stefan". Danach schreibt man einen Punkt. Zum Schluß folgt die gewünschte Variable, also z.B. vname. Bemerkung: Die Anwendung von Strukturen verdeutlicht folgendes Beispiel:

struct PERSON hubert;
struct PERSON stefan;

strcpy (hubert.vname, "Max");
strcpy (stefan.nname,"Mustermann");

Zeiger

Jede Variable steht an einer genau definierten stelle im Speicher. Diese Stelle kann man in erfahrung bringen. Wie man das macht und was das überhaupt für Vorteile bringt, erfahren Sie in diesem Abschnitt. Ein Zeiger ist im Prinzip eine (int-) Variable. In ihr ist jedoch keine Zahl gespeichert, sondern eine "Adresse". Diese gibt eine bestimmte Position im Arbeitsspeicher dar. Hier sehen Sie, wie man einen Zeiger definiert und mit ihm arbeitet:

#include <conio.h>
#include <stdio.h>

void main(void)
{
  int Zahl;
  int *Zeiger;
 
  Zeiger=&Zahl;
  *Zeiger=12;
  
  printf("%d = %d", Zahl, *Zeiger);

Erklärung: Um einen Zeiger anzulegen, gibt man folgendes an: Datentyp, dann einen Stern (*), anschließend den Namen (und einen Strichpunkt). Um einen Zeiger auf eine bestimmte Zahl "zeigen" zu lassen, muß man dem Zeiger die Adresse einer Variablen zuordnen. Das funktioniert mittels dem Adress-Operator & (Zeiger=&Zahl). Jetzt möchten Sie dem Zeiger einen Wert zuweisen. Aber natürlich nicht dem Zeiger, sondern der Variablen, auf die der Zeiger "zeigt". Na, "Zahl=Wert;", werden Sie sagen. Aber wozu dann Zeiger? Na, ja, jednefalls geht das mittels des "Inhalts-Operators" * (*Zeiger=12). Genauso können Sie mit dem Inhaltsoperator Werte abfragen und an printf (und jedes andere Unterprogramm) übergeben.

Zeiger als Parameter

Wenn Sie ein Unterprogramm aufrufen, können Sie diesem Parameter übergeben, aber keine Werte zurückgekommen (außer den Funktionswert bei Funktionen). Dies hat einen guten Grund: beim Aufruf werden nicht die aufgerufenen Parameter benutzt, sondern es werden deren Werte in neue Variablen kopiert. Diese Variablen werden am Ende des Unterprogrammes "zerstört", ohne ihre Werte an die aufrufenden Parameter zu übergeben. Jede Veränderung eines Parameters hat daher keine Auswirkung auf den Parameter. Doch was ist, wenn Sie Parameter in Unterprogrammen verändern möchten? Ganz einfach, Sie verwenden Zeiger. Der Computer legt dann immer noch Kopien an. In dieser Kopie steht aber kein Wert, sondern die Adresse einer Varaiblen. Und auf diese können Sie dann zugreifen. Denken Sie nur an scanf - da übergeben Sie ja auch die Adresse einer Variablen ("&Variable").

#include <conio.h>
#include <stdio.h>

void erhoehe(int *zeiger)
{
  *zeiger=*zeiger+1;
}

void main(void)
{
  int Zahl;
  printf("Zahl eingeben: ");
  scanf("%d",&Zahl);
  erhoehe(&Zahl);
  printf("\nDie erhoehte Zahl lautet: %d\n",Zahl);
  getch();
}

Parameter von main

Das Unterprogramm "main" kann, wie jede andere Funktion, Parameter besitzen. Doch keine selbst gewählten, sondern nur bestimmte. Doch warum braucht main Parameter? Denken Sie einmal an alle Betriebssystembefehle: "dir *.exe", "copy *.* a:" oder "ls -la". All diese Befehle sind aus zwei Teilen aufgebaut: Befehl und Parameter. Und genau diese Parameter können Sie mit den main-Parametern abfragen.

void main (int argc, char *argv[], char* environ[])

Bei "argc" handelt es sich um eine normale int-Variable (engl. "argument count", "Parameter-Zähler"). In ihr steht die Anzahl der übergebenen Parameter. Die Parameter selbst folgen im zweiten Argument, das als Array von Strings übergeben wird. Das dritte Argument ist ein Array mit den Umgebungsvariablen. Seine Länge wird nicht explizit übergeben; nach dem letzten Element steht ein Null-String, also ein String der Länge 0. In dieser Array befindet sich auch der Inhalt der Umgebungsvariablen PATH, die den Suchpfad für ausführbare Programme enthält.

#include <stdio.h>
#include <stdlib.h>

void main (int argc, char *argv[], char * environ[])
{
  int i;

  printf ("Es wurden %d Parameter angegeben", argc);

  for (i=0; i < argc; i++) 
     printf ("Parameter %d: %s\n", i, argv[i]);

  for (i = 0; environ[i] != NULL; ++i) 
     printf ("environ[%d] = %s\n", i, environ[i]);
}
Erklärung
Bei der ersten Ausgabe wird ausgegeben, wie viele Parameter insgesammt angegeben wurden. Dabei gibt immer mindestens einen Parameter, nämlich argc[0]. Dort steht der Name der aufgerufenen Datei selbst. Außerdem ist das letzte gültige Feldelement – wie in C üblich – das Element <tt>argv[argc-1]. In der for-Schleife werden alle Parameter, inklusive ihrer Nummer, ausgegeben. Experimentieren Sie mit den Parametern, um das System zu vertehen!


Autoren

  • Plasma
  • Bernd

Quellen:

  • Kernighan und Ritchie - Buch
  • Christian Wirth , C Tutorial
  • Prof. Dr. J. Dankert Ausführungen

Siehe auch

Weblinks


LiFePO4 Speicher Test