Compilerwarnung (Stufe 2) C4146
Einem vorzeichenlosen Typ wurde ein unärer Operator Minus zugewiesen. Das Ergebnis ist weiterhin vorzeichenlos.
Vorzeichenlose Typen können ausschließlich nicht negative Werte enthalten, sodass eine auf einen vorzeichenlosen Typ angewendete Negation (unärer Operator Minus) in der Regel keinen Sinn ergibt. Sowohl der Operand als auch das Ergebnis sind nicht negativ.
In der Praxis tritt diese Situation ein, wenn der Programmierer versucht, den minimalen Ganzzahlwert, d. h. -2147483648, auszudrücken. Dieser Wert kann nicht mit "-2147483648" dargestellt werden, da der Ausdruck in zwei Schritten verarbeitet wird:
Die Zahl 2147483648 wird ausgewertet. Da größer als der maximale Ganzzahlwert 2147483647, hat 2147483648 nicht den Typ int, sondern unsigned int.
Der unäre Operator Minus wird auf den Wert angewendet. Dies führt zu einem vorzeichenlosen Ergebnis, das zufällig auch den Wert 2147483648 hat.
Der vorzeichenlose Typ des Ergebnisses kann zu unerwartetem Verhalten führen. Falls das Ergebnis in einem Vergleich verwendet wird, sollte dieser, beispielsweise wenn der andere Operand ein int-Typ ist, ohne Vorzeichen durchgeführt werden. Dies ist der Grund dafür, dass das unten aufgeführte Beispielprogramm lediglich eine Zeile ausgibt.
Die erwartete zweite Zeile 1 is greater than the most negative int wird nicht ausgegeben, da ((unsigned int)1) > 2147483648 "false" ist.
Um C4146 zu vermeiden, verwenden Sie INT_MIN aus "limits.h" mit dem Typ signed int.
Beispiel
Im folgenden Beispiel wird C4146 generiert:
// C4146.cpp
// compile with: /W2
#include <stdio.h>
void check(int i)
{
if (i > -2147483648) // C4146
printf_s("%d is greater than the most negative int\n", i);
}
int main()
{
check(-100);
check(1);
}