Acabo de tener un desagradable error en C++. Así que tenía una lista de registros y valores, que están envueltos en una estructura y luego esas estructuras se inicializan en una matriz. Pero luego accidentalmente escribí ()
en lugar de {}
. Aquí hay un código de prueba:Significados de paréntesis en C++ y C
#include <stdio.h>
struct reg_val {
unsigned reg;
unsigned val;
};
struct reg_val faulty_array[] = {
{0x5001, 0xff},
{0x5580, 0x01},
(0x5580, 0x02), //<- THIS LINE IS THE PROBLEM
(0x5589, 0x00), //<- AND THIS LINE
};
struct reg_val good_array[] = {
{0x5001, 0xff},
{0x5580, 0x01},
{0x5580, 0x02},
{0x5589, 0x00},
};
int main()
{
unsigned i;
unsigned faulty_size = sizeof(faulty_array)/sizeof(struct reg_val);
printf("Size of faulty array: %d\n", faulty_size);
for (i = 0; i < faulty_size; ++i) {
printf("faulty reg: %x val: %x\n", faulty_array[i].reg,
faulty_array[i].val);
}
unsigned good_size = sizeof(good_array)/sizeof(struct reg_val);
printf("\nSize of good array: %d\n", good_size);
for (i = 0; i < good_size; ++i) {
printf("good reg: %x val: %x\n", good_array[i].reg,
good_array[i].val);
}
return 0;
}
estoy más familiarizado con C y para mi sorpresa esto todavía se compila con g ++:
$ g++ -Wall array.cc
array.cc:11: warning: left-hand operand of comma has no effect
array.cc:12: warning: left-hand operand of comma has no effect
array.cc:13: warning: missing braces around initializer for ‘reg_val’
$ ./a.out
Size of faulty array: 3
faulty reg: 5001 val: ff
faulty reg: 5580 val: 1
faulty reg: 2 val: 0 <-- the first value gets discarded as mentioned in the compiler warning
Size of good array: 4
good reg: 5001 val: ff
good reg: 5580 val: 1
good reg: 5580 val: 2
good reg: 5589 val: 0
Este código, obviamente, dejar de compilar con un compilador C, ¿cuál es la diferencia en C++ que hace que un compilador de C++ (aunque a regañadientes) acepte este código?
Ah, gracias. Ya veo. Eso tiene sentido. Pero gcc parece no estar muy contento con mi "aplanamiento de llaves". '$ gcc -Wall array.c array.c: 13: advertencia: falta llaves alrededor del inicializador' – Lucas
' gcc' con '-Wall' también te advertirá sobre' if (c = something()) 'porque' sugiere paréntesis alrededor de la asignación utilizada como valor de verdad'. El hecho de que algo esté permitido por ANSI C no significa que no generará una advertencia. –
tiene razón, compila 'gcc -std = c89 -pedantic array.c' sin advertencias. – Lucas