abstract class base { public int age=getNumber(100); static { System.out.println("base static block"); } { System.out.println("base nonstatic block"); } static int sage=getNumber(50); base() { System.out.println(age); System.out.println("base start"); draw();//會調用子類覆蓋后的方法 System.out.println("base end"); } static int getNumber(int base) { System.out.println("base.getNumber int"+base); return base; } public void draw() { System.out.println("base.draw"); } } public class initializeOrder extends base{ public int age=getNumber(1001); private int _radius=getNumber(10); static int sage=getNumber(250); static { System.out.println("subclass static block"); } { System.out.println("subclass nonstatic block"); } initializeOrder(int radius) { _radius=radius; System.out.println(age); System.out.println("initializeOrder initialized"); } public void draw() { System.out.println("initializeOrder.draw "+_radius); } /** *//** * @param args */ public static void main(String[] args) { // TODO Auto-generated method stub new initializeOrder(1000); } }
運行結果:
C:\java>java initializeOrder
base static block
base.getNumber int50
base.getNumber int250
subclass static block
base.getNumber int100
base nonstatic block
100
base start
initializeOrder.draw 0
base end
base.getNumber int1001
base.getNumber int10
subclass nonstatic block
1001
initializeOrder initialized
C:\java>
總結:
1、類只有在使用New調用創建的時候才會被JAVA類裝載器裝入
2、JAVA類首次裝入時,會對靜態成員變量或方法進行一次初始化,但方法不被調用是不會執行的,靜態成員變量和靜態初始化塊級別相同,非靜態成員變量和非靜態初始化塊級別相同。
先初始化父類的靜態代碼--->初始化子類的靜態代碼-->
初始化父類的非靜態代碼--->初始化父類構造函數--->
初始化子類非靜態代碼--->初始化子類構造函數
3、創建類實例時,首先按照父子繼承關系進行初始化
4、類實例創建時候,首先初始化塊部分先執行,然后是構造方法;然后從
本類繼承的子類的初始化塊執行,最后是子類的構造方法
5、類消除時候,首先消除子類部分,再消除父類部分
測試:
package com.javabase;
/**
* <p>Title: 平時小例子練習</p>
*
* <p>Description: 平時小例子練習</p>
*
* <p>Copyright: Copyright (c) 2005</p>
*
* <p>Company: 北京開元東方科技有限公司</p>
*
* @author 樊建強
* @version 1.0
* 測試java類加載初始化順序
*/
abstract class base
{
public int age=getNumber(100);
static
{
System.out.println("base static block");
}
{
System.out.println("base nonstatic block");
}
static int sage=getNumber(50);
base()
{
System.out.println(age);
System.out.println("base start");
draw();//會調用子類覆蓋后的方法,哪批怕寫在游離塊中也會執行子類覆蓋后的方法
System.out.println("base end");
}
static int getNumber(int base)
{
System.out.println("base.getNumber int"+base);
return base;
}
public void draw()
{
System.out.println("base.draw");
}
}
public class JavaBase extends base{
public int age=getNumber(1001);
private int _radius=getNumber(10);
static int sage=getNumber(250);
static
{
System.out.println("subclass static block");
}
{
System.out.println("subclass nonstatic block");
}
JavaBase(int radius)
{
_radius=radius;
System.out.println(age);
System.out.println("initializeOrder initialized");
}
public void draw()
{
System.out.println("initializeOrder.draw "+_radius);
}
/** *//**
* @param args
*/
public static void main(String[] args) {
// TODO Auto-generated method stub
System.out.println("-------------類初始化開始-------------");
new JavaBase(1000);
System.out.println("-------------類初始化結束-------------");
}
}
執行結果:
base static block
base.getNumber int50
base.getNumber int250
subclass static block
-------------類初始化開始-------------
base.getNumber int100
base nonstatic block
initializeOrder.draw 0
100
base start
base end
base.getNumber int1001
base.getNumber int10
subclass nonstatic block
1001
initializeOrder initialized
-------------類初始化結束-------------
更正:static靜態成員變量和靜態初始化是在new對象之前進行的.
后待添加更正***************************************************************
Static詳解:
一、static
請先看下面這段程序: public class Hello{ 看過這段程序,對于大多數學過Java 的從來說,都不陌生。即使沒有學過Java,而學過其它的高級語言,例如C,那你也應該能看懂這段代碼的意思。它只是簡單的輸出“Hello,world”,一點別的用處都沒有,然而,它卻展示了static關鍵字的主要用法。 | |
|
在1處,我們定義了一個靜態的方法名為main,這就意味著告訴Java編譯器,我這個方法不需要創建一個此類的對象即可使用。你還記得你是怎么運行這個程序嗎?一般,我們都是在命令行下,打入如下的命令:
javac Hello.java
java Hello
Hello,world!
這就是你運行的過程,第一行用來編譯Hello.java這個文件,執行完后,如果你查看當前,會發現多了一個Hello.class文件,那就是第一行產生的Java二進制字節碼。第二行就是執行一個Java程序的最普遍做法。執行結果如你所料。在2中,你可能會想,為什么要這樣才能輸出。好,我們來分解一下這條語句。(如果沒有安裝Java文檔,請到Sun的官方網站瀏覽J2SE API)首先,System是位于java.lang包中的一個核心類,如果你查看它的定義,你會發現有這樣一行:public static final PrintStream out;接著再進一步,點擊PrintStream這個超鏈接,在METHOD頁面,你會看到大量定義的方法,查找println,會有這樣一行:
public void println(String x)。
好了,現在你應該明白為什么我們要那樣調用了,out是System的一個靜態變量,所以可以直接使用,而out所屬的類有一個println方法。
靜態方法
通常,在一個類中定義一個方法為static,那就是說,用類名而無需本類的對象即可調用此方法。如下所示:
class Simple{
static void go(){
System.out.println("Go...");
}
}
public class Cal{
public static void main(String[] args){
Simple.go();
}
}
調用一個靜態方法就是“類名.方法名”,靜態方法的使用很簡單如上所示。一般來說,靜態方法常常為應用程序中的其它類提供一些實用工具所用,在Java的類庫中大量的靜態方法正是出于此目的而定義的。
靜態變量
靜態變量與靜態方法類似。所有此類實例共享此靜態變量,也就是說在類裝載時,只分配一塊存儲空間,所有此類的對象都可以操控此塊存儲空間,當然對于final則另當別論了。看下面這段代碼:
class Value{
static int c=0;
static void inc(){
c++;
}
}
class Count{
public static void prt(String s){
System.out.println(s);
}
public static void main(String[] args){
Value v1,v2;
v1=new Value();
v2=new Value();
prt("v1.c="+v1.c+" v2.c="+v2.c);
v1.inc();
prt("v1.c="+v1.c+" v2.c="+v2.c);
}
}
結果如下:
v1.c=0 v2.c=0
v1.c=1 v2.c=1
由此可以證明它們共享一塊存儲區。static變量有點類似于C中的全局變量的概念。值得探討的是靜態變量的初始化問題。我們修改上面的程序:
class Value{
static int c=0;
Value(){
c=15;
}
Value(int i){
c=i;
}
static void inc(){
c++;
}
}
class Count{
public static void prt(String s){
System.out.println(s);
}
Value v=new Value(10);
static Value v1,v2;
static{
prt("v1.c="+v1.c+" v2.c="+v2.c);
v1=new Value(27);
prt("v1.c="+v1.c+" v2.c="+v2.c);
v2=new Value(15);
prt("v1.c="+v1.c+" v2.c="+v2.c);
}
public static void main(String[] args){
Count ct=new Count();
prt("ct.c="+ct.v.c);
prt("v1.c="+v1.c+" v2.c="+v2.c);
v1.inc();
prt("v1.c="+v1.c+" v2.c="+v2.c);
prt("ct.c="+ct.v.c);
}
}
運行結果如下:
v1.c=0 v2.c=0
v1.c=27 v2.c=27
v1.c=15 v2.c=15
ct.c=10
v1.c=10 v2.c=10
v1.c=11 v2.c=11
ct.c=11
這個程序展示了靜態初始化的各種特性。如果你初次接觸Java,結果可能令你吃驚。可能會對static后加大括號感到困惑。首先要告訴你的是,static定義的變量會優先于任何其它非static變量,不論其出現的順序如何。正如在程序中所表現的,雖然v出現在v1和v2的前面,但是結果卻是v1和v2的初始化在v的前面。在static{后面跟著一段代碼,這是用來進行顯式的靜態變量初始化,這段代碼只會初始化一次,且在類被第一次裝載時。如果你能讀懂并理解這段代碼,會幫助你對static關鍵字的認識。在涉及到繼承的時候,會先初始化父類的static變量,然后是子類的,依次類推。
通常一個普通類不允許聲明為靜態的,只有一個內部類才可以。這時這個聲明為靜態的內部類可以直接作為一個普通類來使用,而不需實例一個外部類。如下代碼所示:
public class StaticCls{
public static void main(String[] args){
OuterCls.InnerCls oi=new OuterCls.InnerCls();
}
}
class OuterCls{
public static class InnerCls{
InnerCls(){
System.out.println("InnerCls");
}
}
}
輸出結果會如你所料:
InnerCls