2i
jest gcc
rozszerzeniem zespolonego literału liczby całkowitej, czystej liczby urojonej, która jest dwa razy większa od pierwiastka kwadratowego -1
. To rozszerzenie jest również obsługiwane przez clang
.
Zaskakujące jest to, że kompilacja z gcc 5.4.0
generuje wysłany wynik montażu:
- Kompilując na http://gcc.godbolt.org/# Otrzymuję błąd kompilacji z
gcc
5.3.0 http://gcc.godbolt.org/#
:: error: cannot convert '__complex__ int' to 'int' in return
.
- Wysłany kod asemblera funkcji
foo
jest nieprawidłowy: nie zwraca 0
. Przekształcenie stałej zespolonej liczby całkowitej 2i
na int
powinno zwrócić jej część rzeczywistą 0
.
I odwrotnie, w wersji clang
3.7 kompiluje się bez ostrzeżenia i generuje optymalny kod, ale oczywiście nie taki, jakiego oczekujesz:
foo(int): # @foo(int)
xorl %eax, %eax
retq
Tę składnię można łączyć z innymi przyrostkami w dowolnej kolejności. Kompilacja poniższego kodu clang -Weverything
daje mi odpowiednie ostrzeżenia warning: imaginary constants are a GNU extension [-Wgnu-imaginary-constant]
:
#include <stdio.h>
int main() {
printf("sizeof(2i) = %zd\n", sizeof(2i));
printf("sizeof(2ui) = %zd\n", sizeof(2ui));
printf("sizeof(2li) = %zd\n", sizeof(2li));
printf("sizeof(2lli) = %zd\n", sizeof(2lli));
printf("sizeof(2.i) = %zd\n", sizeof(2.i));
printf("sizeof(2.fi) = %zd\n", sizeof(2.fi));
printf("sizeof(2e0fi) = %zd\n", sizeof(2e0fi));
printf("sizeof(2e0i) = %zd\n", sizeof(2e0i));
printf("sizeof(2il) = %zd\n", sizeof(2il));
printf("sizeof(2ill) = %zd\n", sizeof(2ill));
printf("sizeof(2.if) = %zd\n", sizeof(2.if));
return 0;
}
Tworzy ten wynik w moim środowisku:
sizeof(2i) = 8
sizeof(2ui) = 8
sizeof(2li) = 16
sizeof(2lli) = 16
sizeof(2.i) = 16
sizeof(2.fi) = 8
sizeof(2e0fi) = 8
sizeof(2e0i) = 16
sizeof(2il) = 16
sizeof(2ill) = 16
sizeof(2.if) = 8
Wypróbuj ostatnią za pomocą edytora kolorowania składni ;-)