C Language
Conversioni implicite ed esplicite
Ricerca…
Sintassi
- Conversione esplicita (aka "Casting"): (tipo) espressione
Osservazioni
" Conversione esplicita " è anche comunemente chiamato "casting".
Conversioni intere in chiamate di funzione
Dato che la funzione ha un prototipo corretto, gli interi vengono ampliati per le chiamate alle funzioni secondo le regole della conversione dei numeri interi, C11 6.3.1.3.
6.3.1.3 Numeri interi firmati e non firmati
Quando un valore con tipo intero viene convertito in un altro tipo intero diverso da _Bool, se il valore può essere rappresentato dal nuovo tipo, non viene modificato.Altrimenti, se il nuovo tipo non è firmato, il valore viene convertito aggiungendo o sottraendo ripetutamente un valore superiore al valore massimo che può essere rappresentato nel nuovo tipo finché il valore non si trova nell'intervallo del nuovo tipo.
Altrimenti, il nuovo tipo è firmato e il valore non può essere rappresentato in esso; o il risultato è definito dall'implementazione o viene generato un segnale definito dall'implementazione.
Di solito non si deve troncare un ampio tipo firmato con un tipo firmato più stretto, perché ovviamente i valori non possono essere adattati e non vi è alcun chiaro significato che ciò dovrebbe avere. Lo standard C sopra citato definisce questi casi come "definiti dall'implementazione", cioè non sono portabili.
L'esempio seguente suppone che int
sia a 32 bit di larghezza.
#include <stdio.h>
#include <stdint.h>
void param_u8(uint8_t val) {
printf("%s val is %d\n", __func__, val); /* val is promoted to int */
}
void param_u16(uint16_t val) {
printf("%s val is %d\n", __func__, val); /* val is promoted to int */
}
void param_u32(uint32_t val) {
printf("%s val is %u\n", __func__, val); /* here val fits into unsigned */
}
void param_u64(uint64_t val) {
printf("%s val is " PRI64u "\n", __func__, val); /* Fixed with format string */
}
void param_s8(int8_t val) {
printf("%s val is %d\n", __func__, val); /* val is promoted to int */
}
void param_s16(int16_t val) {
printf("%s val is %d\n", __func__, val); /* val is promoted to int */
}
void param_s32(int32_t val) {
printf("%s val is %d\n", __func__, val); /* val has same width as int */
}
void param_s64(int64_t val) {
printf("%s val is " PRI64d "\n", __func__, val); /* Fixed with format string */
}
int main(void) {
/* Declare integers of various widths */
uint8_t u8 = 127;
uint8_t s64 = INT64_MAX;
/* Integer argument is widened when function parameter is wider */
param_u8(u8); /* param_u8 val is 127 */
param_u16(u8); /* param_u16 val is 127 */
param_u32(u8); /* param_u32 val is 127 */
param_u64(u8); /* param_u64 val is 127 */
param_s8(u8); /* param_s8 val is 127 */
param_s16(u8); /* param_s16 val is 127 */
param_s32(u8); /* param_s32 val is 127 */
param_s64(u8); /* param_s64 val is 127 */
/* Integer argument is truncated when function parameter is narrower */
param_u8(s64); /* param_u8 val is 255 */
param_u16(s64); /* param_u16 val is 65535 */
param_u32(s64); /* param_u32 val is 4294967295 */
param_u64(s64); /* param_u64 val is 9223372036854775807 */
param_s8(s64); /* param_s8 val is implementation defined */
param_s16(s64); /* param_s16 val is implementation defined */
param_s32(s64); /* param_s32 val is implementation defined */
param_s64(s64); /* param_s64 val is 9223372036854775807 */
return 0;
}
Conversioni puntatore nelle chiamate di funzione
Le conversioni del puntatore a void*
sono implicite, ma qualsiasi altra conversione del puntatore deve essere esplicita. Mentre il compilatore consente una conversione esplicita da qualsiasi tipo di puntatore a dati a qualsiasi altro tipo di puntatore a dati, l'accesso a un oggetto tramite un puntatore errato è errato e porta a un comportamento indefinito. L'unico caso in cui questi sono consentiti sono se i tipi sono compatibili o se il puntatore con cui stai guardando l'oggetto è un tipo di carattere.
#include <stdio.h>
void func_voidp(void* voidp) {
printf("%s Address of ptr is %p\n", __func__, voidp);
}
/* Structures have same shape, but not same type */
struct struct_a {
int a;
int b;
} data_a;
struct struct_b {
int a;
int b;
} data_b;
void func_struct_b(struct struct_b* bp) {
printf("%s Address of ptr is %p\n", __func__, (void*) bp);
}
int main(void) {
/* Implicit ptr conversion allowed for void* */
func_voidp(&data_a);
/*
* Explicit ptr conversion for other types
*
* Note that here although the have identical definitions,
* the types are not compatible, and that the this call is
* erroneous and leads to undefined behavior on execution.
*/
func_struct_b((struct struct_b*)&data_a);
/* My output shows: */
/* func_charp Address of ptr is 0x601030 */
/* func_voidp Address of ptr is 0x601030 */
/* func_struct_b Address of ptr is 0x601030 */
return 0;
}