java中static{}语句块详解
原文地址:http://blog.csdn.net/lubiaopan/article/details/4802430 ? ? 感謝原作者!
static{}(即static塊),會(huì)在類被加載的時(shí)候執(zhí)行且僅會(huì)被執(zhí)行一次,一般用來(lái)初始化靜態(tài)變量和調(diào)用靜態(tài)方法,下面我們?cè)敿?xì)的討論一下該語(yǔ)句塊的特性及應(yīng)用。
?
一、在程序的一次執(zhí)行過(guò)程中,static{}語(yǔ)句塊中的內(nèi)容只被執(zhí)行一次,看下面的示例:
示例一
[java]? view plain copy
class?Test{??
????????public?static?int?X=100;??
????public?final?static?int?Y;=200??
????public?Test(){??
????????System.out.println("Test構(gòu)造函數(shù)執(zhí)行");??
????}??
????static{??
????????System.out.println("static語(yǔ)句塊執(zhí)行");??
????}??
????public?static?void?display(){??
????????System.out.println("靜態(tài)方法被執(zhí)行");??
????}??
????public?void?display_1(){??
????????System.out.println("實(shí)例方法被執(zhí)行");??
????}??
}??
public?class?StaticBlockTest{??
????public?static?void?main(String?args[]){??
????????try{??
????????????????Class.forName("Test");?????
????????????????????Class.forName("Test");???
????????}catch(ClassNotFoundException?e){??
????????????e.printStackTrace();??
????????}??
????????????
????}?????
}??
??
結(jié)果:你會(huì)發(fā)現(xiàn)雖然執(zhí)行了兩條Class.forName("Test")語(yǔ)句,但是,只輸出了一條"靜態(tài)方法被執(zhí)行"語(yǔ)句;其實(shí)第二條Class.forName()語(yǔ)句已經(jīng)無(wú)效了,因?yàn)樵谔摂M機(jī)的生命周期中一個(gè)類只被加載一次;又因?yàn)閟tatic{}是伴隨類加載執(zhí)行的,所以,不管你new多少次對(duì)象實(shí)例,static{}都只執(zhí)行一次。 --關(guān)于類加載請(qǐng)看本文的附錄。
?
二、static{}語(yǔ)句塊執(zhí)行的時(shí)機(jī)(其實(shí)就是附錄中類加載的時(shí)機(jī))
?
上面說(shuō)到static{}會(huì)在類被加載的時(shí)候執(zhí)行,我們必須準(zhǔn)確理解類加載的準(zhǔn)確含義,含義如下:
1、用Class.forName()顯示加載的時(shí)候,如上面的示例一;
2、實(shí)例化一個(gè)類的時(shí)候,如將main()函數(shù)的內(nèi)容改為:Test t=new Test();//這種形式其實(shí)和1相比,原理是相同的,都是顯示的加載這個(gè)類,讀者可以驗(yàn)證Test t=new Test();和Test t=(Test)Class.forName().newInstance();這兩條語(yǔ)句效果相同。
3、調(diào)用類的靜態(tài)方法的時(shí)候,如將main()函數(shù)的內(nèi)容改為:Test.display();
4、調(diào)用類的靜態(tài)變量的時(shí)候,如將main()函數(shù)的內(nèi)容改為:System.out.println(Test.X);
?
總體來(lái)說(shuō)就這四種情況,但是我們特別需要注意一下兩點(diǎn):
1、調(diào)用類的靜態(tài)常量的時(shí)候,是不會(huì)加載類的,即不會(huì)執(zhí)行static{}語(yǔ)句塊,讀者可以自己驗(yàn)證一下(將main()函數(shù)的內(nèi)容改為System.out.println(Test.Y);),你會(huì)發(fā)現(xiàn)程序只輸出了一個(gè)200;(這是java虛擬機(jī)的規(guī)定,當(dāng)訪問(wèn)類的靜態(tài)常量時(shí),如果編譯器可以計(jì)算出常量的值,則不會(huì)加載類,否則會(huì)加載類)
2、用Class.forName()形式的時(shí)候,我們也可以自己設(shè)定要不要加載類,如將Class.forName("Test")改為?Class.forName("Test",false,StaticBlockTest.class.getClassLoader()),你會(huì)發(fā)現(xiàn)程序什么都沒(méi)有輸出,即Test沒(méi)有被加載,static{}沒(méi)有被執(zhí)行。
?
三、static{}語(yǔ)句塊的執(zhí)行次序
?
1、當(dāng)一個(gè)類中有多個(gè)static{}的時(shí)候,按照static{}的定義順序,從前往后執(zhí)行;
2、先執(zhí)行完static{}語(yǔ)句塊的內(nèi)容,才會(huì)執(zhí)行調(diào)用語(yǔ)句;
示例二
public class TestStatic{
??? static{
??????? System.out.println(1);
??? }
??? static {
??????? System.out.println(2);
??? }
??? static {
??????? System.out.println(3);
??? }
??? public static void main(String args[]){
??????? System.out.println(5);
??? }
??? static {
??????? System.out.println(4);
??? }
}
結(jié)果:程序會(huì)輸出1,2,3,4,5
3、如果靜態(tài)變量在定義的時(shí)候就賦給了初值(如 static int X=100),那么賦值操作也是在類加載的時(shí)候完成的,并且當(dāng)一個(gè)類中既有static{}又有static變量的時(shí)候,同樣遵循“先定義先執(zhí)行”的原則;
示例三
?class Test{
?public static int X=300;
?static{
??System.out.println(X);
??X=200;
??System.out.println(X);
?}
}
public class StaticBlockTest{
?public static void main(String args[]){
??System.out.println(Test.X);
?}
}
結(jié)果:程序會(huì)依次輸出300,200,200,先執(zhí)行完X=300,再執(zhí)行static{}語(yǔ)句塊。
?
四、static{}語(yǔ)句塊應(yīng)用
?
1、JDBC中的應(yīng)用
熟悉JDBC的讀者應(yīng)該知道,java中有一個(gè)DriverManager類,用于管理各種數(shù)據(jù)庫(kù)驅(qū)動(dòng)程序、建立新的數(shù)據(jù)庫(kù)連接。DriverManager類包含一些列Drivers類,這些Drivers類必須通過(guò)調(diào)用DriverManager的registerDriver()方法來(lái)對(duì)自己進(jìn)行注冊(cè),那么注冊(cè)是什么時(shí)候發(fā)生的呢?下面會(huì)給出答案:
所有Drivers類都必須包含有一個(gè)靜態(tài)方法,利用這個(gè)靜態(tài)方法可以創(chuàng)建該類的實(shí)例,然后在加載該實(shí)例時(shí)向DriverManage類進(jìn)行注冊(cè)。我們經(jīng)常用Class.forName()對(duì)驅(qū)動(dòng)程序進(jìn)行加載,那么注冊(cè)就發(fā)生在這條語(yǔ)句的執(zhí)行過(guò)程中,前面說(shuō)的Drivers的靜態(tài)方法是放在static{}中的,當(dāng)對(duì)驅(qū)動(dòng)程序進(jìn)行加載的時(shí)候,會(huì)執(zhí)行該static{},便完成了注冊(cè)。
?
2、hibernate中的應(yīng)用
hibernate中的SessionFactory是一個(gè)重量級(jí)的類,創(chuàng)建該類的對(duì)象實(shí)例會(huì)耗費(fèi)比較多的系統(tǒng)資源,如果每次需要時(shí)都創(chuàng)建一個(gè)該類的實(shí)例,顯然會(huì)降低程序的執(zhí)行效率,所以經(jīng)常將對(duì)該類的實(shí)例化放在一個(gè)static{}中,只需第一次調(diào)用時(shí)執(zhí)行,提高程序的執(zhí)行效率,如下:
static {
??? ?try {
???configuration.configure(configFile);
???sessionFactory = configuration.buildSessionFactory();
??} catch (Exception e) {
???System.err.println("%%%% Error Creating SessionFactory %%%%");
???e.printStackTrace();
??}
??? }
?
五、附錄
類加載:Java命令的作用是啟動(dòng)虛擬機(jī),虛擬機(jī)通過(guò)輸入流,從磁盤上將字節(jié)碼文件(.class文件)中的內(nèi)容讀入虛擬機(jī),并保存起來(lái)的過(guò)程就是類加載。
?
類加載特性?:
??????*在虛擬機(jī)的生命周期中一個(gè)類只被加載一次。
???? ?*類加載的原則:延遲加載,能少加載就少加載,因?yàn)樘摂M機(jī)的空間是有限的。
???? ?*類加載的時(shí)機(jī):
????? 1)第一次創(chuàng)建對(duì)象要加載類.
????? 2)調(diào)用靜態(tài)方法時(shí)要加載類,訪問(wèn)靜態(tài)屬性時(shí)會(huì)加載類。
????? 3)加載子類時(shí)必定會(huì)先加載父類。
????? 4)創(chuàng)建對(duì)象引用不加載類.
????? 5) 子類調(diào)用父類的靜態(tài)方法時(shí)
????????? (1)當(dāng)子類沒(méi)有覆蓋父類的靜態(tài)方法時(shí),只加載父類,不加載子類
????????? (2)當(dāng)子類有覆蓋父類的靜態(tài)方法時(shí),既加載父類,又加載子類
??????6)訪問(wèn)靜態(tài)常量,如果編譯器可以計(jì)算出常量的值,則不會(huì)加載類,例如:public static final int a =123;否則會(huì)加載類,例如:public static final int a = math.PI。
轉(zhuǎn)載于:https://my.oschina.net/liangzhenghui/blog/194559
總結(jié)
以上是生活随笔為你收集整理的java中static{}语句块详解的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問(wèn)題。
- 上一篇: 二维数组的应用
- 下一篇: identifier of an ins