Quel est l'avantage d'utiliser uint8_t
over unsigned char
en C?
Je sais que sur presque tous les systèmes, ce uint8_t
n'est qu'un typedef unsigned char
, alors pourquoi l'utiliser?
Il documente votre intention - vous stockerez de petits nombres, plutôt qu'un caractère.
De plus, cela semble plus agréable si vous utilisez d'autres types de caractères tels que uint16_t
ou int32_t
.
unsigned char
ousigned char
documenter explicitement l'intention aussi, car sans fioritureschar
est ce qui montre que vous travaillez avec des personnages.unsigned
étaitunsigned int
par définition?char
semble impliquer un caractère, alors que dans le contexte d'une chaîne UTF8, il peut ne s'agir que d'un octet d'un caractère multi-octets. L'utilisation de uint8_t pourrait indiquer clairement qu'il ne faut pas s'attendre à un caractère à chaque position - en d'autres termes que chaque élément de la chaîne / du tableau est un entier arbitraire sur lequel il ne faut pas faire d'hypothèses sémantiques. Bien sûr, tous les programmeurs C le savent, mais cela peut pousser les débutants à poser les bonnes questions.Juste pour être pédant, certains systèmes peuvent ne pas avoir un type 8 bits. Selon Wikipedia :
Il
uint8_t
n'est donc pas garanti d'exister, bien qu'il le soit pour toutes les plates-formes où 8 bits = 1 octet. Certaines plates-formes intégrées peuvent être différentes, mais cela devient très rare. Certains systèmes peuvent définir deschar
types sur 16 bits, auquel cas il n'y aura probablement aucun type sur 8 bits.À part ce problème (mineur), la réponse de @Mark Ransom est la meilleure à mon avis. Utilisez celui qui montre le plus clairement pourquoi vous utilisez les données.
En outre, je suppose que vous vouliez dire
uint8_t
(le typedef standard de C99 fourni dans l'en-stdint.h
tête) plutôt queuint_8
(ne faisant partie d'aucune norme).la source
uint8_t
(ou le taper typef pour cela). En effet, le type 8 bits aurait des bits inutilisés dans la représentation de stockage, quiuint8_t
ne doivent pas avoir.typedef unsigned integer type uint8_t; // optional
Donc, en substance, une bibliothèque conforme à la norme C ++ n'est pas nécessaire du tout pour définir uint8_t (voir le commentaire // optionnel )Le but est d'écrire du code indépendant de l'implémentation.
unsigned char
n'est pas garanti comme étant de type 8 bits.uint8_t
est (si disponible).la source
sizeof(unsigned char)
retournera1
pour 1 octet. mais si un système char et int ont la même taille, par exemple, 16 bits, alorssizeof(int)
ils reviendront également1
Comme vous l'avez dit, " presque tous les systèmes".
char
est probablement l'un des moins susceptibles de changer, mais une fois que vous commencez à utiliseruint16_t
et vos amis, vous utilisezuint8_t
mieux les mélanges, et peut même faire partie d'une norme de codage.la source
D'après mon expérience, il y a deux endroits où nous voulons utiliser uint8_t pour signifier 8 bits (et uint16_t, etc.) et où nous pouvons avoir des champs inférieurs à 8 bits. Les deux endroits sont où l'espace compte et nous avons souvent besoin de regarder un vidage brut des données lors du débogage et de pouvoir déterminer rapidement ce qu'il représente.
Le premier concerne les protocoles RF, en particulier dans les systèmes à bande étroite. Dans cet environnement, nous devrons peut-être regrouper autant d'informations que possible dans un seul message. Le second est dans le stockage flash où nous pouvons avoir un espace très limité (comme dans les systèmes embarqués). Dans les deux cas, nous pouvons utiliser une structure de données compressée dans laquelle le compilateur se chargera de l'emballage et du déballage pour nous:
La méthode que vous utilisez dépend de votre compilateur. Vous devrez peut-être également prendre en charge plusieurs compilateurs différents avec les mêmes fichiers d'en-tête. Cela se produit dans les systèmes embarqués où les périphériques et les serveurs peuvent être complètement différents - par exemple, vous pouvez avoir un périphérique ARM qui communique avec un serveur Linux x86.
Il y a quelques mises en garde concernant l'utilisation de structures compactes. Le plus gros problème est que vous devez éviter de déréférencer l'adresse d'un membre. Sur les systèmes avec des mots alignés sur plusieurs octets, cela peut entraîner une exception mal alignée - et un coredump.
Certaines personnes s'inquiéteront également des performances et soutiendront que l'utilisation de ces structures compressées ralentira votre système. Il est vrai que, dans les coulisses, le compilateur ajoute du code pour accéder aux membres de données non alignés. Vous pouvez le voir en regardant le code assembleur dans votre IDE.
Mais comme les structures compressées sont les plus utiles pour la communication et le stockage des données, les données peuvent être extraites dans une représentation non compressée lorsque vous travaillez avec elles en mémoire. Normalement, nous n'avons pas besoin de travailler avec le paquet de données entier en mémoire de toute façon.
Voici quelques discussions pertinentes:
pragma pack (1) ni __attribute__ ((aligné (1))) fonctionne
Le pack __attribute __ ((emballé)) / #pragma de gcc est-il dangereux?
http://solidsmoke.blogspot.ca/2010/07/woes-of-structure-packing-pragma-pack.html
la source
Il y a peu. Du point de vue de la portabilité,
char
ne peut pas être inférieur à 8 bits, et rien ne peut être inférieur àchar
, donc si une implémentation C donnée a un type entier 8 bits non signé, ce sera le caschar
. Alternativement, il peut ne pas en avoir du tout, à quel point lestypedef
astuces sont sans objet.Il pourrait être utilisé pour mieux documenter votre code dans un sens qu'il est clair que vous avez besoin d'octets 8 bits et rien d'autre. Mais dans la pratique, c'est une attente raisonnable pratiquement n'importe où déjà (il existe des plates-formes DSP sur lesquelles ce n'est pas vrai, mais les chances que votre code s'exécute là-bas sont minces, et vous pourriez tout aussi bien vous tromper en utilisant une assertion statique en haut de votre programme sur une telle plateforme).
la source
unsigned char
de pouvoir contenir des valeurs entre 0 et 255. Si vous pouvez le faire en 4 bits, mon chapeau est à vous.uint8_t
à l'implémentation. Je me demande, les compilateurs pour DSP avec des caractères 16 bits implémentent-ils généralementuint8_t
ou non?#include <stdint.h>
et d'utiliseruint8_t
. Si la plate-forme l'a, elle vous la donnera. Si la plate-forme ne l'a pas, votre programme ne sera pas compilé et la raison sera claire et simple.C'est très important par exemple lorsque vous écrivez un analyseur de réseau. les en-têtes de paquet sont définis par la spécification du protocole, et non par le fonctionnement du compilateur C d'une plate-forme particulière.
la source
Sur presque tous les systèmes, j'ai rencontré uint8_t == char non signé, mais ce n'est pas garanti par la norme C. Si vous essayez d'écrire du code portable et que la taille de la mémoire est exacte, utilisez uint8_t. Sinon, utilisez un caractère non signé.
la source
uint8_t
correspond toujours à la plage et à la taille deunsigned char
et au remplissage (aucun) lorsqu'ilunsigned char
est à 8 bits. Quandunsigned char
n'est pas 8 bits,uint8_t
n'existe pas.unsigned char
est 8 bits, estuint8_t
garanti d'être un detypedef
ceux - ci et non untypedef
d'un type entier non signé étendu ?unsigned char/signed char/char
le type le plus petit - pas plus petit que 8 bits.unsigned char
n'a pas de rembourrage. Pouruint8_t
être, il doit être de 8 bits, pas de remplissage, exister à cause d'une implémentation de type entier fourni: correspondant aux exigences minimales deunsigned char
. Quant à "... garanti d'être un typedef ..." ressemble à une bonne question à poster.