Compile Time Constants in Java

Compile Time Constants in Java

In this tutorial we’ll get a basic understanding of Compile Time Constants in Java programming language. This is not an exhaustive tutorial about compile time constants, but this should be enough for most people. We’ll understand what compile time constants are and how are they special. First lets go through the general rules regarding compile time constants.
  • They must be declared final
  • They are of primitive data types or String
  • They must be initialized with their declaration.
  • Their value must be constant expression.
If you don’t understand all of this, don’t worry, we’ll go through these rules now.
Final variables of primitive data types and Strings can be compile time constants. No other type of variables are compile time constants, not even the wrapper classes. They must also be initialized with their declaration, otherwise they’ll not be compile time constants. Lets take a few examples. All of the following are compile time constants,
final int i = 10;
final int j = 20, k = 30;
final String s = "Hello";
final float f = 10.5f;
The following are not compile time constants,
final Integer i = 10;    //not a primitive or String
int j = 10;              //not final
final int k;             //not initialized with declaration
= 10;
final int l, m = l = 20; //both l and m are not compile time constants
The last example might not be intuitive. We are initializing l and m in the same statement as their declaration, but they will not be compile time constants. Compile time constants have to be initialized right with their declaration and you can’t use a different variable in the middle of their declaration.
The last rule about compile time constants is that the compiler must be able to deduce their value. So their value can be any expression which contains literals and other compile time constants. Lets have a look at some valid values for compile time constants,
final int i = 10 * 20;  //uses only literals
final int j = i;        //uses another compile time constant
final int k = i * 20;   //mixing compile time constant and literal
If you use a variable in the expression to assign value to a final variable, then the compiler won’t be able to figure out the value of the final variable, so it will not be a compile time constant. The final variables given below are not compile time constants, because their value is not a constant expression,
int i = 10;
final int j = i;                 //using a variable
final int k = Math.round(10.2);  //method call


Compile Time Constants in Java! What’s So special About Them?

Now that we know what compile time constants are, we’ll now see why are they special. As the name implies, compile time constants get special treatment by the compiler. The first example of the special treatment is that they are implicit downcasted (just like literals). If you assign an int variable with value 10 to a short variable, you’ll get an error, but if you assign a int which is a compile time constant with the value 10 to a short variable, that will compile fine. Here is an example,
int a = 10;
short s1 = a;   //error
final int b = 10;
short s2 = b;   //fine
Since the compiler knows the value of compile time constants, the compler knows that the value of b is 10, so the compiler knows that the value of b is within the range of short data type. You can also use compile time constants as case values in switch-case statements as shown in the following example,
final int a = 10;
int b = 10;
switch(b) {
    case a:
        //some code
    case 20:
        //some code
}
But remember, case values cannot be String or floating point numbers, so you can’t use compile time constants of float, double or String data types as case values.
Both the behaviors that we observed above i.e. implicit downcast and case labels are because of a common reason. Wherever you use a compile time constant, the compiler replaces their use with their actual value. So if we write this code,
final int i = 10;
short j = i;
System.out.println(i);
After compilation it will become,
final int i = 10;
short j = 10;System.out.println(10);
As you can see, at both the places where we used i, the compiler replaced it with its value 10. This is the reason why they can be used as case values and that is why they are implicitly downcasted. This replacing of value has serious implications in some cases.
Suppose we have two classes in two different files like this,
public class ConstantClass {
    public static final int MY_CONST = 10;
}
public class Main {
    public static void main(String[] args) {
        System.out.println(ConstantClass.MY_CONST);
    }
}
Now if we compile both classes and run Main class, the output will be 10. Since MY_CONST is a compile time constant, so the compiler replaces it with its value in the main method. Now we go and change the value of MY_CONST to 20 and compile only ConstantClass class. Then we run the Main class (without compiling it again), the output will still be 10 i.e. the value of MY_CONST when Main class was compiled. We’ll have to recompile Main class to see the new value of MY_CONSTas the output.
One more special treatment that compile time constants receive is that you can declare static compile time constants inside non-static inner classes. As you might know, non-static inner classes cannot have static members. But you are allowed to add static compile time constants to non-static inner classes.


Comments