Question

public class Test {
    public static void main(String[] args) {

        String s = null;
        String s1 = null;
        Integer i = null;
        Integer i1 = null;

        System.out.println(s+i);
        System.out.println(i+s);
        System.out.println(s+s1);

        try {
            System.out.println(i+i1);
        } catch (NullPointerException np) {         
            System.out.print("NullPointerException");       
        }
    }

}

The question is simple - why do I receive a NullPointerException only at the last line?

Was it helpful?

Solution

Your code makes use of two different additive operators. The first three lines use string concatenation, whereas the last one uses numeric addition.

String concatenation is well-defined to turn null into "null":

  • If the reference is null, it is converted to the string "null" (four ASCII characters n, u, l, l).

Hence there is no NPE.

Adding two Integer objects together requires them to be unboxed. This results in the null reference being dereferenced, which leads to the NPE:

  • If r is null, unboxing conversion throws a NullPointerException

OTHER TIPS

Notice that first three + operator usages involve string concatenation. Only the last one is the actual numeric sum. When string concatenation is involved (where s variable is involved), Java compiler uses some clever trick to improve performance. It replaces + operator with StringBuilder. For example your first line is translated to:

StringBuilder tmp = new StringBuilder();
tmp.append(s);
tmp.append(i);
System.out.println(tmp);

StringBuilder is null-friendly so no matter what you pass as an argument, it nicely replaces it with "null" string.

The situation is different in the last line. There you reference two Integer objects. The only that the JVM can do here is to unbox them (i.intValue()) and do the actual computation. Unboxing of null causes NullPointerException.

The concatenation (+ operator) of anything with a String results in a String. Every operand is first converted to a String (either using toString(), or using the value "null"), then concatenated.

But the last operation involves Integers only, so the previous rules don't apply. Instead of a concatenation, it does an addition. But to add two Integers, it converts objects (Integers) to primitive values (int), which is not possible if the Integer is null. That's why you get a NullPointerException.

Seems to be a case of auto-unboxing. If you've got two Integers first and second, then the result of their addition will be first.intValue() + second.intValue(). Since both of them are null, that results in an NPE.

Have a look at the bytecode for you program. You will notice that your null object is being passed as a parameter to the PrintStream.print() method. The source code for the print() method uses String.valueOf() as shown below:

public static String valueOf(Object obj) {
    return (obj == null) ? "null" : obj.toString();
}

It should be noted that the answer to this question should have been obvious from the output:

C:\Temp>java Test
nullnull
nullnull
nullnull
NullPointerException
Licensed under: CC-BY-SA with attribution
Not affiliated with StackOverflow
scroll top