Programmation C/Types avancés
Structures
modifierstruct ma_structure {
type1 champ1;
type2 champ2;
...
typeN champN;
} var1, var2, ..., varM;
Déclare une structure (ou enregistrement) ma_structure composé de N champs, champ1 de type type1, champ2 de type type2, etc. On déclare, par la même occasion, M variables de type struct ma_structure.
Accès aux champs
modifierL'accès aux champs d'une structure se fait avec un point :
struct complexe {
int reel;
int imaginaire;
} c;
c.reel = 1;
c.imaginaire = 2;
Initialisation
modifierIl y a plusieurs façons d'initialiser une variable de type structure :
- En initialisant les champs un à un :
struct t_mastruct { char ch; int nb; float pi; }; struct t_mastruct variable; variable.ch = 'a'; variable.nb = 12345; variable.pi = 3.141592;
Cette façon est néanmoins pénible lorsqu'il y a beaucoup de champs.
-
À la déclaration de la variable :
struct t_mastruct { char ch; int nb; float pi; }; struct t_mastruct variable = { 'a', 12345, 3.141592 };
Les valeurs des champs sont assignés dans l'ordre où ils sont déclarés. S'il manque des initialisations, les champs seront initialisés à 0. L'inconvénient, c'est qu'on doit connaitre l'ordre où sont déclarés les champs, ce qui peut être tout aussi pénible à retrouver, et peut causer des plantages lorsque la définition de la structure est modifiée.
-
Une nouveauté du C99 permet d'initialiser certains champs à la déclaration de la variable, en les nommant :
struct t_mastruct { char ch; int nb; float pi; }; struct t_mastruct variable = { .pi = 3.141592, .ch = 'a', .nb = 12345 };
Les champs non initialisés auront des valeurs indéfinies. Il devient possible de ne pas respecter l'ordonnancement des champs.
Manipulation
modifierLa seule opération prise en charge par le langage est la copie, lors des affectations ou des passages de paramètres à des fonctions. Toutes les autres opérations sont à la charge du programmeur, notamment la comparaison d'égalité (cf. section suivante).
Alignement et bourrage (padding)
modifierIl s'agit d'un concept relativement avancé, mais qu'il est bien de connaitre pour agir en connaissance de cause. Lorsqu'on déclare une structure, on pourrait naïvement croire que les champs se suivent les uns à la suite des autres en mémoire. Or, il arrive souvent que des octets soient intercalés entre les champs.
Considérez la structure suivante:
struct ma_structure {
char champ1; /* 8 bits */
int champ2; /* 32 bits */
char champ3; /* 8 bits */
};
On pourrait penser que cette structure occupera 6 octets en mémoire, et pourtant, sur une bonne partie des compilateurs, on obtiendrait une taille plus proche des 12 octets.
En fait, les compilateurs insèrent quasiment toujours des octets entre les champs pour pouvoir les aligner sur des adresses qui correspondent à des mots machines. Cela est dû à une limitation de la plupart des processeurs, qui ne peuvent lire des « mots » de plus d'un octet que s'ils sont alignés sur un certain adressage (alors qu'il n'y a pas de contrainte particulière pour lire un seul octet, si le type char
est défini sur 8bit).
En se représentant la mémoire comme un tableau continu, on peut tracer le dessin suivant:
| bloc N | bloc N + 1 | bloc N + 2 | ---+---------------+---------------+---------------+--- | a | b | c | d | e | f | g | h | i | j | k | l | ---+---------------+---------------+---------------+---
Les cases a
, b
, c
, ... représentent des octets, et les blocs des sections de 32 bits. Si on suppose qu'une variable de type ma_structure
doive être placée en mémoire à partir du bloc numéro N, alors un compilateur pourra, pour des raisons de performance, placer champ1
en a
, champ2
de e
à h
, et champ3
en i
. Cela permettrait en effet d'accéder simplement à champ2
: le processeur fournit des instructions permettant de lire ou d'écrire directement le bloc N + 1. Dans ce cas, les octets de b
à d
ne sont pas utilisés; on dit alors que ce sont des octets de bourrage (ou padding en anglais).
Un autre compilateur (ou le même, appelé avec des options différentes) peut aussi placer champ2
de b
à e
, et champ3
en f
, pour optimiser l'utilisation mémoire. Mais alors il devra générer un code plus complexe lors des accès à champ2
, le matériel ne lui permettant pas d'accéder en une seule instruction aux 4 octets b
à e
.
En fait il faut garder à l'esprit que toutes les variables suivent cette contrainte: aussi bien les variables locales aux fonctions, les champs de structures, les paramètres de fonctions, etc.
L'existence d'octets de bourrage ainsi que leur nombre sont non seulement dépendants de l'architecture, mais aussi du compilateur. Cela dit, il est toujours possible de connaître la « distance » (offset) d'un champ par rapport au début de la structure, et ce, de manière portable. Pour cela il existe une macro, déclarée dans l'entête <stddef.h>
:
size_t offsetof(type, champ);
La valeur renvoyée est le nombre de char
(i.e. d'octets la plupart du temps), entre le début de la structure et celui du champ. Le premier argument attendu est bel et bien le type de la structure et non une variable de ce type, ni un pointeur vers une variable de ce type. Pour s'en souvenir, il suffit de savoir que beaucoup d'implémentations d'offsetof
utilisent une arithmétique de ce genre:
size_t distance = (size_t) &((type *)NULL)->champ;
Si type
était un pointeur, il faudrait faire un déréférencement supplémentaire (ou éviter l'étoile dans la macro). À noter que, même si cette macro peut s'avérer contraignante (notamment lorsqu'on ne dispose que de type pointeur), il est quand même préférable de l'utiliser pour des raisons de portabilité.
Voici un petit exemple d'utilisation de cette macro:
#include <stddef.h>
#include <stdio.h>
struct ma_structure {
char champ1;
int champ2;
char champ3;
};
int main(void)
{
/* en C99 */
/*
printf("L'offset de 'champ2' vaut %zu.\n",
offsetof(struct ma_structure, champ2));
*/
/* en C90 */
printf("L'offset de 'champ2' vaut %lu.\n",
(unsigned long) offsetof(struct ma_structure, champ2));
return 0;
}
Sur une architecture 32 bits, vous obtiendrez très certainement la réponse:
L'offset de 'champ2' vaut 4.
En fait toute cette section était pour souligner le fait qu'il est difficilement portable de comparer les structures comme des blocs binaires (via la fonction memcmp
par exemple), car il n'y a aucune garantie que ces octets de bourrage soient initialisés à une certaine valeur. De la même manière, il est sage de prendre quelques précautions avant de transférer cette structure à l'extérieur du programme (comme un fichier, un tube de communication ou une socket IP). En général, il préférable de traiter la structure champ par champ, pour ce genre d'opérations.
Mais comme rien n'est toujours totalement négatif, les conséquences du bourrage offre énormément de souplesse : imaginons 2 structures:
struct str {
char *string;
};
struct str_ok {
char *string;
size_t len;
};
quelle que soit la structure utilisé, on peut avec l'adresse et la cast (struct str *) accéder à la chaine.
int main (void)
{
struct str s = { "chaine" };
struct str_ok sok = { "chaine", 0 };
void *p = &sok;
if (((struct str *)p)->string)
{
sok.len = strlen(sok.string);
printf("%s\n", ((struct str *)p)->string);
}
/* en revanche: Peut-être très dangereux ! */
p = &s;
printf("%s\n",((struct str_ok *)p)->string);
return 0;
}
Pointeurs vers structures
modifierIl est (bien entendu) possible de déclarer des variables de type pointeur vers structure :
struct ma_struct * ma_variable;
Comme pour tout pointeur, on doit allouer de la mémoire pour la variable avant de l'utiliser :
ma_variable = malloc( sizeof(struct ma_struct) );
L'accès aux champs peut se faire comme pour une variable de type structure « normale » :
(* ma_variable).champ
Ce cas de figure est en fait tellement fréquent qu'il existe un raccourci pour l'accès aux champs d'un pointeur vers structure :
ma_variable->champ
Unions
modifierUne union et un enregistrement se déclarent de manière identique :
union type_union
{
type1 champ1;
type2 champ2;
/* ... */
typeN champN;
};
/* Déclaration de variables */
union type_union var1, var2, /* ... */ varM;
Toutefois, à la différence d'un enregistrement, les N champs d'une instance de cette union occupent le même emplacement en mémoire. Modifier l'un des champ modifie donc tous les champs de l'union. Typiquement, une union s'utilise lorsqu'un enregistrement peut occuper plusieurs fonctions bien distinctes et que chaque fonction ne requiert pas l'utilisation de tous les champs.
Voici un exemple ultra classique d'utilisation des unions, qui provient en fait de la gestion des événements du système X-Window, très répandu sur les systèmes Unix :
union _XEvent
{
int type;
XAnyEvent xany;
XKeyEvent xkey;
XButtonEvent xbutton;
XMotionEvent xmotion;
XCrossingEvent xcrossing;
XFocusChangeEvent xfocus;
XExposeEvent xexpose;
/* ... */
XErrorEvent xerror;
XKeymapEvent xkeymap;
long pad[24];
};
La déclaration a juste été un peu raccourcie pour rester lisible. Les types X*Event
de l'union sont en fait des structures contenant des champs spécifiques au message. Par exemple, si type
vaut ButtonPress
, seules les valeurs du champ xbutton
seront significatives. Parmi ces champs, il y a button
qui permet de savoir quel bouton de la souris à été pressé :
XEvent ev;
XNextEvent(display, &ev);
switch (ev.type) {
case ButtonPress:
printf("Le bouton %d a été pressé.\n", ev.xbutton.button);
break;
default:
printf("Message type %d\n", ev.type);
}
En offrant une souplesse nécessaire à la manipulation de l'objet, la gestion de la mémoire en C peut aider à réduire les erreurs de programmation lié à la structure de la mémoire (erreur de segmentation/segfault/"Argh!!!"):
#include <stdio.h>
#include <string.h>
struct s1 {
char *string;
} s1;
struct s2 {
char *string;
size_t len;
int ______;
int setnull;
} s2;
union str {
struct s2 s_;
struct s1 _s;
}str;
typedef void * STR_s1;
typedef void * STR_s2;
STR_s2
string_analyse(STR_s1 *s)
{
static union str us, init_us = {{NULL,0,0,0}};
memcpy((void *)&us, (void *)&init_us, sizeof(union str));
if (!((struct s1 *)s)->string)
{
return &us._s;
}
us.s_.len = strlen(((struct s1 *)s)->string);
if (!us.s_.len)
{
us.s_.setnull = 1;
return &us._s;
}
us.s_.string = ((struct s1 *)s)->string;
return &us.s_;
}
int main(void)
{
struct s1 s = {NULL}, s_ = {"salut"}, s__ = {""};
void *p;
p = string_analyse((void *)&s);
printf("null? %s,%lu,%i\n",
((struct s2 *)p)->string, ((struct s2 *)p)->len, ((struct s2 *)p)->setnull);
p = string_analyse((void *)&s_);
printf("null? %s,%lu,%i\n",
((struct s2 *)p)->string, ((struct s2 *)p)->len, ((struct s2 *)p)->setnull);
p = string_analyse((void *)&s__);
printf("null? %s,%lu,%i\n",
((struct s2 *)p)->string, ((struct s2 *)p)->len, ((struct s2 *)p)->setnull);
printf("Réalisé en toute sécurité...\n"
"Même pour un petit vaisseau serial killer du monde d'en haut.\n");
return 0;
}
Tout en permettant l'optimisation:
struct s1{
size_t len;
char *str;
} s1;
union s {
struct s1 dbl[2];
void *perform[5];
struct s1 simple;
};
void * | void * | void * | void * | void * |
size_t | char * | size_t | char * | --- |
size_t | char * | ------ | --- | |
n oct | n oct | n oct | n oct | 2int=n oct |
ce qui signifie que:
int main(void)
{
const char *end[2] = {", ", "\n"};
union s init = {{{5,"Salut"}, {3,"toi"}}}, us;
void *p;
int s1s = sizeof(struct s1);
memset((void *)&us, 0,40);
memcpy((void *)&us,
(void *)&init,
2 * s1s);
for (p = &us;
((union s *)p)->perform[0] != (void *)0;
p += s1s)
printf("%s%s",
(char *)((union s *)p)->perform[1],
end[
!((char *)((union s *)p)->perform[2])]
);
return 0;
}
Définitions de synonymes de types (typedef)
modifierLe langage C offre un mécanisme assez pratique pour définir des synonymes de types. Il s'agit du mot-clé typedef
.
typedef un_type synonyme_du_type;
Contrairement aux langages à typage fort comme le C++, le C se base sur les types atomiques pour décider de la compatibilité entre deux types. Dit plus simplement, la définition de nouveaux types est plus un mécanisme d'alias qu'une réelle définition de type. Les deux types sont effectivement parfaitement interchangeables. À la limite on pourrait presque avoir les mêmes fonctionnalités en utilisant le préprocesseur C, bien qu'avec ce dernier vous aurez certainement beaucoup de mal à sortir de tous les pièges qui vous seront tendus.
Quelques exemples
modifiertypedef unsigned char octet;
typedef double matrice4_4[4][4];
typedef struct ma_structure * ma_struct;
typedef void (*gestionnaire_t)( int );
/* Utilisation */
octet nombre = 255;
matrice4_4 identite = { {1,0,0,0}, {0,1,0,0}, {0,0,1,0}, {0,0,0,1} };
ma_struct pointeur = NULL;
gestionnaire_t fonction = NULL;
Ce mot clé est souvent utilisé conjointement avec la déclaration des structures, pour ne pas devoir écrire à chaque fois le mot clé struct
. Elle permet aussi de grandement simplifier les prototypes de fonctions qui prennent des pointeurs sur des fonctions en argument, ou retournent de tels types. Il est conseillé dans de tels cas de définir un type synonyme, plutôt que de l'écrire in extenso dans la déclaration du prototype. Considérez les deux déclarations :
/* Déclaration confuse */
void (*fonction(int, void (*)(int)))(int);
/* Déclaration claire avec typedef */
typedef void (*handler_t)(int);
handler_t fonction(int, handler_t);
Les vétérans auront reconnu le prototype imbitable de l'appel système signal()
, qui permet de rediriger les signaux (interruption, alarme périodique, erreur de segmentation, division par zéro, ...).
Énumérations
modifierenum nom_enum { val1, val2, ..., valN };
Les symboles val1, val2, ..., valN pourront être utilisés littéralement dans la suite du programme. Ces symboles sont en fait remplacés par des entiers lors de la compilation. La numérotation commençant par défaut à 0, s'incrémentant à chaque déclaration. Dans l'exemple ci-dessus, val1 vaudrait 0, val2 1 et valN N-1.
On peut changer à tout moment la valeur d'un symbole, en affectant au symbole, la valeur constante voulue (la numérotation recommençant à ce nouvel indice). Par exemple :
enum Booleen { Vrai = 1, Faux = 0 };
/* Pour l'utiliser */
enum Booleen variable = Faux;
Ce qui est assez pénible en fait, puisqu'il faut à chaque fois se souvenir que le type Booleen
est dérivé d'une énumération. Il est préférable de simplifier les déclarations, grâce à l'instruction typedef :
typedef enum { Faux, Vrai } Booleen;
/* Pour l'utiliser */
Booleen variable = Faux;
Type incomplet
modifierPour garantir un certain degré d'encapsulation, il peut être intéressant de masquer le contenu d'un type complexe, pour éviter les usages trop « optimisés » de ce type. Pour cela, le langage C permet de déclarer un type sans indiquer explicitement son contenu.
struct ma_structure;
/* Plus loin dans le code */
struct ma_structure * nouvelle = alloue_objet();
Les différents champs de la structure n'étant pas connus, le compilateur ne saura donc pas combien de mémoire allouer. On ne peut donc utiliser les types incomplets qu'en tant que pointeur. C'est pourquoi, il est pratique d'utiliser l'instruction typedef
pour alléger les écritures :
typedef struct ma_structure * ma_struct;
/* Plus loin dans le code */
ma_struct nouvelle = alloue_objet();
Cette construction est relativement simple à comprendre, et proche d'une conception objet. À noter que le nouveau type défini par l'instruction typedef
peut très bien avoir le même nom que la structure. C'est juste pour éviter les ambiguités, qu'un nom différent a été choisi dans l'exemple.
Un autre cas de figure relativement classique, où les types incomplets sont très pratiques, ce sont les structures s'auto-référençant, comme les listes chainées, les arbres, etc.
struct liste
{
struct liste * suivant;
struct liste * precedant;
void * element;
};
Ou de manière encore plus tordue, plusieurs types ayant des références croisées :
struct type_a;
struct type_b;
struct type_a
{
struct type_a * champ1;
struct type_b * champ2;
int champ3;
};
struct type_b
{
struct type_a * ref;
void * element;
};