Unary operators are operators that take exactly one argument. Unary operators may appear in a unary expression:
The unary plus and minus operators, a Boolean negation operator (!), a bitwise negation operator (~), and the cast construct comprise the unary operators in Java. The unary operators are equal in precedence and are evaluated from right to left.
References Order of Operations; Postfix Increment/Decrement Operators; Prefix Increment/Decrement Operators; Primary Expressions; Type 3
The unary plus operator (+) can appear as part of a unary expression. The operator does no explicit computation; it produces the same pure value that is produced by its operand. However, the unary + operator may perform a type conversion on its operand. The type of the operand must be an arithmetic data type, or a compile-time error occurs. If the type of the operand is byte, short, or char, the unary + operator produces an int value; otherwise the operator produces a value of the same type as its operand.
References Arithmetic Types
The unary minus operator (-) can appear as part of a unary expression. The type of the operand of the unary - operator must be an arithmetic data type, or a compile-time error occurs. The operator produces a pure value that is the arithmetic negation (i.e., additive inverse) of the value of its operand.
The unary - operator may perform a type conversion. If the type of the operand is byte, short, or char, the operation converts the operand to an int before computing the value's arithmetic negation and producing an int value. Otherwise, unary - produces a value of the same type as its operand.
For integer data types, the unary - operator produces a value equivalent to subtracting its operand from zero. There are, however, negative values for which the unary - operator cannot produce a positive value; in these cases it produces the same negative value as its operand. This behavior results from the two's complement representation Java uses for integer values. The magnitude of the most negative number that can be represented using two's complement notation cannot be represented as a positive number. No exception is thrown when the unary - operator is given a value that cannot be negated. However, you can detect this situation by explicitly testing for these special values. The most negative int value is available as the predefined constant Integer.MIN_VALUE and the most negative long value is available as the predefined constant Long.MIN_VALUE.
For floating-point data types, the unary - operator changes the sign of its operand from + to - or from - to +, for both regular values, positive and negative zero, and positive and negative infinity. The only case where this is not true occurs when the operand is not-a-number (NaN). Given the value NaN, the unary - operator produces NaN.
References Arithmetic Types; Integer; Long
The Boolean negation operator (!) may appear as part of a unary expression. The type of the operand of the ! operator must be boolean, or a compile-time error occurs. If the value of its operand is false, the ! operator produces the pure boolean value true. If the value of its operand is true, the operator produces the pure boolean value false.
Here is an example that uses the Boolean negation operator:
public void paint(Graphics g) { if (!loaded) { //The next 2 lines are executed if loaded is false g.drawString("Loading data", 25, 25); return; } g.drawImage(img, 25, 25, this); }
References Boolean Type
The bitwise negation operator (~) may appear as part of a unary expression. The type of the operand of the ~ operator must be an integer data type, or a compile-time error occurs. The ~ operator may perform a type conversion before it performs its computation. If the type of the operand is byte, short, or char, the operator converts its operand to int before producing a value. Otherwise the ~ operator produces a value of the same type as its operand.
After type conversion, the bitwise negation operator produces a pure value that is the bitwise complement of its operand. In other words, if a bit in the converted operand contains a one, the corresponding bit in the result contains a zero. If a bit in the converted operand contains a zero, the corresponding bit in the result contains a one.
Here's an example that shows the use of the bitwise negation operator:
// zero low order four bits int getNibble(int x) { return x & ~0xf; }
References Integer types
A Type enclosed in parentheses specifies a type cast operation. A cast may appear as part of a unary expression. A cast operation always produces a pure value of the specified type, by converting its operand to that type if necessary. This is different from a type cast in C/C++, which can produce garbage if it is given a pointer to a data type different than that implied by the pointer's declaration. If the actual data type of the operand of a cast cannot be guaranteed at compile-time, the Java compiler must produce code to check the type of the operand at runtime. In Java, any value that gets past all of the type-checking done on a cast is guaranteed to be compatible with the type specified by the cast.
A cast can convert between certain primitive types. A cast between object reference types never alters the type or content of the object, but may alter the type of the reference to the object.
Because it is not possible to convert between all types, some cast operations are permitted and others are not. Here are the rules governing casts:
Object o = "ABC"; String s = (String)o; // This is okay Double d = (Double)o; // Throws an exception
The cast of o to String is fine because o is really a reference to a String object. The cast of o to Double throws an exception at runtime because the object that o references is not an instance of Double.
interface Weber { double flux(double x); } class B {} final class C {} class D implements Weber { public double flux(double x) { return Math.PI*x*x; } } class Intercast { public void main(String[] argv) { B b = new B(); C c = new C(); D d = new D(); Weber w; w = (Weber)b; // Throws an exception w = (Weber)c; // Compiler complains w = (Weber)d; // Okay, D implements Weber } }
The cast of b to Weber is fine with the compiler because the class B might have a subclass that implements Weber. At runtime, however, this cast throws an exception because B does not implement Weber. The cast of c to Weber produces an error message from the compiler, as the C class does not implement Weber. Because C is final, it will not have any subclasses and therefore there is no possibility of c containing a reference to an object that implements the Weber interface. The cast of d to Weber is fine because the D class implements the Weber interface.
Here is an example to illustrate these points:
interface Weber { double flux(double x); } class B {} final class C {} class D implements Weber { public double flux(double x) { return Math.PI*x*x; } } class Intercast { public void doit(Weber w) { B b = (B)w; // May throw an exception C c = (C)w; // Compiler complains D d = (D)w; // Okay } }
The cast of w to class B is fine with the compiler even though B does not implement Weber. The compiler lets it pass because B might have a subclass that implements Weber and w could contain a reference to that class. However, at runtime, the cast will throw an exception if the object actually referenced is not an instance of B or a subclass of B. The cast of w to class C produces an error message from the compiler. C does not implement Weber and C cannot have any subclasses because it is final; any object that implements Weber cannot be an instance of C. The cast of w to class D is fine at compile-time because D implements Weber. At runtime, if w references an object that is not an instance of D, a ClassCastException is thrown.
Here is an example to illustrate these points:
interface Weber { double flux(double x); } interface Dyn { double squeeze(); } interface Press extends Dyn { double squeeze(double theta); } class D implements Press { public double squeeze() { return Math.PI; } public double squeeze(double theta) { return Math.PI*Math.sin(theta); } } class Interinter { public static void doit(D d) { Dyn dyn = d; // Okay Weber w = (Weber)d; // May throw exception } }
The assignment of d to dyn works because d is of class D, D implements Press, and Press extends Dyn. Therefore, d refers to an object that implements Dyn and we have assignment compatibility. The compiler lets the cast of d to Weber pass because there may be a subclass of D that implements Weber. At runtime, the cast will throw an exception if D does not implement Weber.
Any cast operation not covered by the preceding rules is not allowed and the Java compiler issues an error message.
References Arithmetic Types; Array Types; Boolean Type; Class Types; Interface Types; Runtime exceptions