I'm testing with this code:
public class TestNull {
public void leftComparison(String s) {
if (s == null);
}
public void rightComparison(String s) {
if (null == s);
}
}
I compiled it with javac 1.8.0_05, and then inspected the bytecode:
public class TestNull {
....
public void leftComparison(java.lang.String);
Code:
0: aload_1
1: ifnonnull 4
4: return
public void rightComparison(java.lang.String);
Code:
0: aconst_null
1: aload_1
2: if_acmpne 5
5: return
}
Apparently, leftComparison
is compiled to push and pop 1 variable on the stack while rightComparison
pushes and pops 2. I speculate that leftComparison
is therefore slightly more efficient than rightComparison
?
I'm wondering why the compiler doesn't rewrite the code of rightComparison
? In my opinion, the two comparisons should be semantically equivalent, right?
The Java bytecode compiler does very little in the way of optimization. The serious optimization work is almost all done by the JIT compiler.
I'm wondering why the compiler doesn't rewrite the code of rightComparison?
Because there's no point in rewriting it. The JIT compiler should be able to deal with both of them, and most likely generates optimal (native) code for both versions. (You can check this if you are interested. There are ways to see the native code generated by the JIT compiler.)
(Also see @codenheim's answer for a more technical explanation.)
In my opinion, the two comparisons should be semantically equivalent, right?
That is correct ... but it doesn't mean that the bytecode compiler is obligated to generate the same bytecode sequences for both versions.
The real lesson here is that the bytecode sequences generated by the bytecode compiler tell you very little about how your code is actually going to perform. Any performance conclusions that you might draw from reading the bytecodes are highly suspect.