java是多线程的,所以类或者接口的初始化要求仔细的同步,因为同时某个别的线程可能正在试图初始化同一个类或者接口,也有可能作为类或者接口初始化的一部分,要求递归地初始化该类或接口。
<script language="Javascript" type="text/javascript"> </script>
要了解类的初始化过程,首先要知道Class对象的状态,class对象的状态有以下几种:
1、被检验和准备,但是没有被初始化;
2、正在被某个特殊的线程T初始化;
3、被完整地初始化并且可以使用;
4、处于错误状态,可能由于检验或者准备失败,或者由于试图初始化且失败了。
初始化的过程如下:
1、同步代表类或接口的class对象的初始化,包括一直等待到当前线程可以获得那个对象的锁;
2、如果某个别的线程正在初始化这个类或接口,则对这个class对象wait,当前线程醒来时,重复这一步骤;
3、如果当前线程正在对类或者接口初始化,则这一定是初始化的一个递归要求,释放class对象上的锁并正常地结束;
4、如果类或接口已被初始化,则不再需要进一步动作,释放class对象上的锁并正常地结束;
5、如果对象处于错误状态,则不可能初始化,释放class对象上的锁并抛出NoClassDefFoundError;
6、否则,记录对象正在被当前线程初始化的事实,并释放class对象上的锁;
7、接着,如果对象代表一个类而非接口并且该类的超类尚未被初始化,则对超类递归地执行这一过程,如果需要,首先检验并准备这个超类,如果因一个抛出的异常E使用超类初始化突然结束,则锁定这个class对象,把它标为错误的,通知所有正在等待的线程,释放锁,并且突然结束,抛出与初始化该超类所引起的相同的异常;
8、接着,除了final static变量和接口的其值为编译期常数的域首先初始化外,按文本顺序执行类变量的初始化器和类的静态初始化函数,或者接口的域初始化器,就好象它们是一个单个的块;
9、如果初始化函数的执行正常地结束,则锁定这个class对象,把它标为被完整地初始化,通知所有正在等待的线程,释放锁并正常地结束该过程;
10、否则,初始化函数一定是通过抛出某个异常E突然结束,如果E的类不是Error或其子类之一,则用E作为参数创建类ExceptionInInitializerError的一个新实例,并在以后的步骤中用这个对象代替E,但是如果因发生一个OutOfMemoryError而不能创建ExceptionInInitailizerError的新实例,则在以后步骤中用一个OutOfMemoryError对象代替E;
11、锁定这个class对象,把它标为错误的,通知所有正在等待的线程,释放锁,并且带着原因E或者在前面步骤中确定的E的替换突然结束结束这个过程。