This is a follow-up of Java static initialization - part 1. Part 1 went through §12.4.1 When Initialization Occurs of Java language specification.
This post will go throught the rest of 12.4 chapter starting with the section "12.4.2 Detailed Initialization Procedure". As name suggests this section provides detailed description of how a class or interface is initialized. It starts with rationale.
12.4.2 Detailed Initialization Procedure
Because the Java programming language is multithreaded, initialization of a class or interface requires careful synchronization, since some other thread may be trying to initialize the same class or interface at the same time. There is also the possibility that initialization of a class or interface may be requested recursively as part of the initialization of that class or interface; for example, a variable initializer in class A might invoke a method of an unrelated class B, which might in turn invoke a method of class A.
After explanation why this level of detail is needed it describes the procedure itself. This procedure can be rewritten in this Java pseudo-code (comments refer to procedure points):
void initializeClass(Clazz clazz) { boolean repeat = false; while (repeat) { repeat = false; synchronized (clazz) { // 1. InitializationStatus status = clazz.getInitializationStatus(); switch (status) { case PREPARED: //6. clazz.setInitStatus(InitializationStatus.INITIALIZING); clazz.setInitThread(Thread.currentThread()); break; case INITIALIZING: if (clazz.getInitThread() == Thread.currentThread()) { // 3. return; } else { // 2. try { clazz.wait(); } catch (InterruptedException e) { Thread.currentThread().interrupt(); throw new NoClassDefFoundError(); } repeat = true; } break; case INITIALIZED: // 4. return; case ERRONEOUS: // 5. throw new NoClassDefFoundError(); } } } try { Clazz superClass = clazz.getSuperclass(); // 7. if (superClass != null) { initializeClass(superClass); } } catch(Throwable t) { synchronized (clazz){ clazz.setInitStatus(InitializationStatus.ERRONEOUS); clazz.notifyAll(); } throw t; } try { clazz.getClassLoader().desiredAssertionStatus(clazz.getName()); // 8. clazz.<clinit>(); // 9. synchronized (clazz){ // 10. clazz.setInitStatus(InitializationStatus.INITIALIZED); clazz.notifyAll(); } } catch(Throwable t) { if (!(t instanceof Error)){ // 11. try{ t = new ExceptionInInitializerError(t); } catch (OutOfMemoryError oome){ t = oome; } } synchronized (clazz){ // 12. clazz.setInitStatus(InitializationStatus.ERRONEOUS); clazz.notifyAll(); } throw t; } }
The implementation of the Java virtual machine is responsible for taking care of synchronization and recursive initialization by using the following procedure. It assumes that theClass
object has already been verified and prepared, and that theClass
object contains state that indicates one of four situations:
- This
Class
object is verified and prepared but not initialized.- This
Class
object is being initialized by some particular thread T.- This
Class
object is fully initialized and ready for use.- This
Class
object is in an erroneous state, perhaps because initialization was attempted and failed.The procedure for initializing a class or interface is then as follows:
- Synchronize (§14.19) on the
Class
object that represents the class or interface to be initialized. This involves waiting until the current thread can obtain the lock for that object (§17.1).- If initialization is in progress for the class or interface by some other thread, then
wait
on thisClass
object (which temporarily releases the lock). When the current thread awakens from thewait
, repeat this step.- If initialization is in progress for the class or interface by the current thread, then this must be a recursive request for initialization. Release the lock on the
Class
object and complete normally.- If the class or interface has already been initialized, then no further action is required. Release the lock on the
Class
object and complete normally.- If the
Class
object is in an erroneous state, then initialization is not possible. Release the lock on theClass
object and throw aNoClassDefFoundError
.- Otherwise, record the fact that initialization of the
Class
object is now in progress by the current thread and release the lock on theClass
object.- Next, if the
Class
object represents a class rather than an interface, and the superclass of this class has not yet been initialized, then recursively perform this entire procedure for the superclass. If necessary, verify and prepare the superclass first. If the initialization of the superclass completes abruptly because of a thrown exception, then lock thisClass
object, label it erroneous, notify all waiting threads, release the lock, and complete abruptly, throwing the same exception that resulted from initializing the superclass.- Next, determine whether assertions are enabled (§14.10) for this class by querying its defining class loader.
- Next, execute either the class variable initializers and static initializers of the class, or the field initializers of the interface, in textual order, as though they were a single block, except that
final
class variables and fields of interfaces whose values are compile-time constants are initialized first (§8.3.2.1, §9.3.1, §13.4.9).- If the execution of the initializers completes normally, then lock this
Class
object, label it fully initialized, notify all waiting threads, release the lock, and complete this procedure normally.- Otherwise, the initializers must have completed abruptly by throwing some exception E. If the class of E is not
Error
or one of its subclasses, then create a new instance of the classExceptionInInitializerError
, with E as the argument, and use this object in place of E in the following step. But if a new instance ofExceptionInInitializerError
cannot be created because anOutOfMemoryError
occurs, then instead use anOutOfMemoryError
object in place of E in the following step.- Lock the
Class
object, label it erroneous, notify all waiting threads, release the lock, and complete this procedure abruptly with reason E or its replacement as determined in the previous step.(Due to a flaw in some early implementations, a exception during class initialization was ignored, rather than causing an
ExceptionInInitializerError
as described here.)
As mentioned above this detail description allows us to reason about static initialization scenarios including circular dependencies.
Let's have 2 classes A
and B
:
class A { static int Y = B.Y; static int X = 10; } class B { static int X = A.X; static int Y = 25; }
It seems quite straightforward what values these static fields are going to have - A.X
and B.X
will be initialized to 10 and A.Y
and B.Y
will be initialized to 25. But it is not so (obviously - otherwise it would not be worth mentioning). When running this program:
public class TestCircularInitialization { public static void main(String[] args) { int x = A.X; System.out.println("X = " + B.X); System.out.println("Y = " + A.Y); } }
the output is:
X = 0 Y = 25
So what happens. int x = A.X
will start A
initialization. Initialization will try to execute line 2 in the class A
. That will start B
initialization and executing line 7. And that will cause another try to initialize A
. But A
is in the INITIALIZING
state and the initializing thread is the same as the current one (point 3. of the procedure). Thread returns to line 7. and sets B.X
with current value of A.X
- the default value 0. Then it executes line 8 and ends B
initialization where B.X = 0
and B.Y = 25
. A
initialization continues and A.Y
is set to 25
. And then A.X
is set to 10
at last.
Interesting thing happens when TestCircularInitialization
is modified a bit - int x = B.X
:
public class TestCircularInitialization { public static void main(String[] args) { int x = B.X; System.out.println("X = " + B.X); System.out.println("Y = " + A.Y); } }
the output is then:
X = 10 Y = 0
The explanation is equivalent to the previous example.
The example demonstrated why circular dependency is dangerous. The real behaviour is not obvious and depends on the execution path. The fix in this case would be reordering of static fields:
class A { static int X = 10; static int Y = B.Y; } class B { static int Y = 25; static int X = A.X; }
In this form the real behaviour matches the expected behaviour. The best solution would be to avoid circular dependency altogether though.
The specification declares Class
object is used for synchronization of class initialization state but it is not necessary so in reality. (Note: Following examples are JVM specific. Sun JDK 1.6.0_24 is used in the examples below.)
Let's run the following program:
class Foo { static { System.out.println("Foo initialized"); } } public class TestStaticInitLockingBasic { public static void main(String[] args) throws InterruptedException { synchronized (Foo.class) { System.out.println("Foo locked"); new Thread(new Runnable() { public void run() { new Foo(); } }).start(); Thread.sleep(2000); System.out.println("Unlocking Foo"); } } }
This program locks on Foo.class
object at first then it fires second thread. 1st thread then sleeps for 2s. 2nd thread cause Foo
static initialization. At that moment 2nd thread should wait for 1st thread to unlock Foo.class
and then continue with initialization. So at the end the console should contain:
Foo locked Unlocking Foo Foo initialized
But program prints:
Foo locked Foo initialized Unlocking Foo
It proves JVM does not use Class
object as prescribed in specification. Now let's modify program a bit:
class Foo { static { synchronized (Foo.class) { System.out.println("Foo initialized"); } } } public class TestStaticInitLocking { public static void main(String[] args) throws InterruptedException { synchronized (Foo.class) { System.out.println("Foo locked"); new Thread(new Runnable() { public void run() { new Foo(); } }).start(); Thread.sleep(2000); System.out.println("Main thread woke up - creating Foo instance"); new Foo(); System.out.println("Unlocking Foo"); } } }
There are 2 differences from previous code - Foo
's static initialization has additional synchronization on Foo.class
and main thread is creating the new Foo
instance. Program will print:
Foo locked Main thread woke up - creating Foo instance
and then program hangs. The program is using only a single lock and JVM is expected to use a synchronization for static initialization. It means JVM is using some lock other than Foo.class
and program hangs because the threads are locking Foo.class
and JVM's lock in different ordering. Another interesting thing is that thread dump does not detect any deadlock. This means this internal JVM's lock is not using standard JVM's lock mechanisms.
Existing JVM implementation apparently breaks JLS specification but this situation should be officially fixed soon. Surprisingly it won't happen by JVM implementation change but by JVM specification change. A new draft version of JVM specification (JSR‑000924 Java Virtual Machine Specification Java SE 7 Maintenance Review Draft 3) includes updated version of initialization procedure. This version already don't insist on using Class
object for synchronization. The Draft changelog comments on this change:
The initialization procedure for a class or interface relaxes the requirement to lock on the user-visible Class
object of the class or interface being initialized. Few JVM implementations actually lock on this object. Most use an internal lock invisible to user code, and this practice is now permitted by the initialization procedure.
There is the quote from the draft - the discussed part is highlighted.
Because the Java virtual machine is multithreaded, initialization of a class or interface requires careful synchronization, since some other thread may be trying to initialize the same class or interface at the same time. There is also the possibility that initialization of a class or interface may be requested recursively as part of the initialization of that class or interface. The implementation of the Java virtual machine is responsible for taking care of synchronization and recursive initialization by using the following procedure. It assumes that the Class object has already been verified and prepared, and that the Class object contains state that indicates one of four situations:
- This Class object is verified and prepared but not initialized.
- This Class object is being initialized by some particular thread.
- This Class object is fully initialized and ready for use.
- This Class object is in an erroneous state, perhaps because initialization was attempted and failed.
For each class or interface C, there is a unique initialization lock LC. The mapping from C to LC is left to the discretion of the Java virtual machine implementation. The procedure for initializing C is then as follows:
- Synchronize on the initialization lock, LC, for C. This involves waiting until the current thread can acquire LC.
- If the Class object for C indicates that initialization is in progress for C by some other thread, then release LC and block the current thread until informed that the in-progress initialization has completed, at which time repeat this procedure.
- If the Class object for C indicates that initialization is in progress for C by the current thread, then this must be a recursive request for initialization. Release LC and complete normally.
- If the Class object for C indicates that C has already been initialized, then no further action is required. Release LC and complete normally.
- If the Class object for C is in an erroneous state, then initialization is not possible. Release LC and throw a NoClassDefFoundError.
- Otherwise, record the fact that initialization of the Class object for C is in progress by the current thread, and release LC. Then, initialize each final static field of C with the constant value in its ConstantValue attribute (§4.7.2), in the order the fields appear in the ClassFile.
- Next, if C is a class rather than an interface, and its superclass SC has not yet been initialized, then recursively perform this entire procedure for SC. If necessary, verify and prepare SC first. If the initialization of SC completes abruptly because of a thrown exception, then acquire LC, label the Class object for C as erroneous, notify all waiting threads, release LC, and complete abruptly, throwing the same exception that resulted from initializing SC.
- Next, determine whether assertions are enabled for C by querying its defining class loader.
- Next, execute the class or interface initialization method of C.
- If the execution of the class or interface initialization method completes normally, then acquire LC, label the Class object for C as fully initialized, notify all waiting threads, release LC, and complete this procedure normally.
- Otherwise, the class or interface initialization method must have completed abruptly by throwing some exception E . If the class of E is not Error or one of its subclasses, then create a new instance of the class ExceptionInInitializerError with E as the argument, and use this object in place of E in the following step. If a new instance of ExceptionInInitializerError cannot be created because an OutOfMemoryError occurs, then use an OutOfMemoryError object in place of E in the following step.
- Acquire LC, label the Class object for C as erroneous, notify all waiting threads, release LC, and complete this procedure abruptly with reason E or its replacement as determined in the previous step.
An implementation may optimize this procedure by eliding the lock acquisition in step 1 (and release in step 4/5) when it can determine that the initialization of the class has already completed, provided that, in terms of the memory model, all happens-before orderings (JLS3 §17.4.5) that would exist if the lock were acquired, still exist when the optimization is performed.
Now back to static initialization specification. It continues with the last section 12.4.3.
12.4.3 Initialization: Implications for Code Generation
Code generators need to preserve the points of possible initialization of a class or interface, inserting an invocation of the initialization procedure just described. If this initialization procedure completes normally and the
Class
object is fully initialized and ready for use, then the invocation of the initialization procedure is no longer necessary and it may be eliminated from the code-for example, by patching it out or otherwise regenerating the code.Compile-time analysis may, in some cases, be able to eliminate many of the checks that a type has been initialized from the generated code, if an initialization order for a group of related types can be determined. Such analysis must, however, fully account for concurrency and for the fact that initialization code is unrestricted.
This section allows JVM to eliminate synchronization on initialized classes. Please note the quote from JavaSE7 Draft provides the means for the same optimization at the end.
Overall the static initialization has some tricky areas but these can be usually avoided. It is just another area where "keep it simple, ..." simply works.
Thanks for this article! :)
ReplyDeleteThe last example with Foo and TestStaticInitLocking classes is just BRILLIANT! Thanks a lot for the explanation of differences between Java5/6 and Java7 language specification concerning static initialization.