在前面幾個章節中我們常常使用到System.out.println(),那麼它是什麼呢?
println()是一個方法(Method),而System是系統類別(Class),out是標準輸出物件(Object)。這句話的用法是呼叫系統類別System中的標準輸出物件out中的方法println()。
Java方法是語句的集合,它們在一起執行一個功能。
方法是解決一類問題的步驟的有序組合
方法包含於類別或物件中
方法在程式中被創建,在其他地方被引用
一般情況下,定義一個方法包含以下語法:
修飾符傳回值類型方法名(參數類型參數名){
…
方法體...
return回傳值;
}
方法包含一個方法頭和一個方法體。下面是一個方法的所有部分:
修飾符:修飾符,這是可選的,告訴編譯器如何呼叫方法。定義了該方法的存取類型。
傳回值類型:方法可能會傳回值。 returnValueType是方法傳回值的資料型別。有些方法執行所需的操作,但沒有傳回值。在這種情況下,returnValueType就是關鍵字void 。
方法名:是方法的實際名稱。方法名和參數表共同構成方法簽章。
參數類型:參數像是一個佔位符。當方法被呼叫時,傳遞值給參數。這個值稱為實參或變數。參數列表是指方法的參數類型、順序和參數的個數。參數是可選的,方法可以不包含任何參數。
方法體:方法體包含具體的語句,定義方法的功能。
如:
publicstaticintage(intbirthday){...}
參數可以有多個:
staticfloatinterest(floatprincipal,intyear){...}
注意:在一些其它語言中方法指涉過程和函數。一個傳回非void類型回傳值的方法稱為函數;一個傳回void類型回傳值的方法叫做過程。
下面的方法包含2個參數num1和num2,它會傳回這兩個參數的最大值。
/**傳回兩個整數變數資料的較大值*/
publicstaticintmax(intnum1,intnum2){
intresult;
if(num1>num2){
result=num1;
}else{
result=num2;
}
returnresult;
}
Java支援兩種呼叫方法的方式,根據方法是否傳回值來選擇。
當程式呼叫一個方法時,程式的控制權交給了被呼叫的方法。當被呼叫方法的回傳語句執行或到達方法體閉括號時交還控制權給程式。
當方法傳回一個值的時候,方法呼叫通常被當作一個值。例如:
intlarger=max(30,40);
如果方法回傳值是void,方法呼叫一定是一條語句。例如,方法println傳回void。下面的呼叫是個語句:
System.out.println("WelcometoJava!");
下面的範例示範如何定義一個方法,以及如何呼叫它:
publicclassTestMax{
/**主方法*/
publicstaticvoidmain(String[]args){
inti=5;
intj=2;
intk=max(i,j);
System.out.println("Themaximumbetween"+i+
"and"+j+"is"+k);
}
/**傳回兩個整數變數較大的值*/
publicstaticintmax(intnum1,intnum2){
intresult;
if(num1>num2){
result=num1;
}else{
result=num2;
}
returnresult;
}
}
以上實例編譯運行結果如下:
Themaximumbetween5and2is5
這個程式包含main方法和max方法。 Main方法是被JVM呼叫的,除此之外,main方法和其它方法沒什麼區別。
main方法的頭部是不變的,如例子所示,帶有修飾符public和static,返回void類型值,方法名字是main,此外帶一個String[]類型參數。 String[]表示參數是字串陣列。
本節說明如何宣告和呼叫一個void方法。
下面的範例聲明了一個名為printGrade的方法,並且呼叫它來列印給定的分數。
publicclassTestVoidMethod{
publicstaticvoidmain(String[]args){
printGrade(78.5);
}
publicstaticvoidprintGrade(doublescore){
if(score>=90.0){
System.out.println('A');
}
elseif(score>=80.0){
System.out.println('B');
}
elseif(score>=70.0){
System.out.println('C');
}
elseif(score>=60.0){
System.out.println('D');
}
else{
System.out.println('F');
}
}
}
以上實例編譯運行結果如下:
C
這裡printGrade方法是一個void型別方法,它不回傳值。
一個void方法的呼叫一定是一個語句。 所以,它被在main方法第三行以語句形式呼叫。就像任何以分號結束的語句一樣。
呼叫一個方法時候需要提供參數,你必須按照參數清單指定的順序提供。
例如,下面的方法連續n次列印一個訊息:
publicstaticvoidnPrintln(Stringmessage,intn){
for(inti=0;i<n;i++)
System.out.println(message);
}
下面的範例示範按值傳遞的效果。
該程式建立一個方法,該方法用於交換兩個變數。
publicclassTestPassByValue{
publicstaticvoidmain(String[]args){
intnum1=1;
intnum2=2;
System.out.println("Beforeswapmethod,num1is"+
num1+"andnum2is"+num2);
//呼叫swap方法swap(num1,num2);
System.out.println("Afterswapmethod,num1is"+
num1+"andnum2is"+num2);
}
/**交換兩個變數的方法*/
publicstaticvoidswap(intn1,intn2){
System.out.println("tInsidetheswapmethod");
System.out.println("ttBeforeswappingn1is"+n1
+"n2is"+n2);
//交換n1與n2的值inttemp=n1;
n1=n2;
n2=temp;
System.out.println("ttAfterswappingn1is"+n1
+"n2is"+n2);
}
}
以上實例編譯運行結果如下:
Beforeswapmethod,num1is1andnum2is2
Insidetheswapmethod
Beforeswappingn1is1n2is2
Afterswappingn1is2n2is1
Afterswapmethod,num1is1andnum2is2
傳遞兩個參數呼叫swap方法。有趣的是,方法被呼叫後,實參的值並沒有改變。
上面使用的max方法僅適用於int型資料。但如果你想得到兩個浮點數型資料的最大值呢?
解決方法是建立另一個有相同名字但參數不同的方法,如下列程式碼所示:
publicstaticdoublemax(doublenum1,doublenum2){
if(num1>num2){
returnnum1;
}else{
returnnum2;
}
}
如果你呼叫max方法時傳遞的是int型參數,則int型參數的max方法就會被呼叫;
如果傳遞的是double型參數,則double類型的max方法體會被調用,這叫做方法重載;
就是說一個類別的兩個方法擁有相同的名字,但是有不同的參數列表。
Java編譯器根據方法簽章判斷哪個方法應該被呼叫。
方法重載可以讓程式更清晰易讀。執行密切相關任務的方法應該使用相同的名字。
重載的方法必須擁有不同的參數清單。你不能僅僅依據修飾符或返回類型的不同來重載方法。
變數的範圍是程式中該變數可以被引用的部分。
方法內定義的變數稱為局部變數。
局部變數的作用範圍從聲明開始,直到包含它的區塊結束。
局部變數必須聲明才可以使用。
方法的參數範圍涵蓋整個方法。參數實際上是一個局部變數。
for循環的初始化部分宣告的變量,其作用範圍在整個循環。
但循環體內聲明的變數其適用範圍是從它聲明到循環體結束。它包含如下所示的變數宣告:
你可以在一個方法裡,不同的非嵌套區塊中多次聲明一個具有相同的名稱局部變量,但你不能在巢狀區塊內兩次聲明局部變數。
有時候你希望執行一個程式時候再傳遞給它訊息。這要靠傳遞命令列參數給main()函數實作。
命令列參數是在執行程式時候緊跟在程式名字後面的資訊。
下面的程式列印所有的命令列參數:
publicclassCommandLine{
publicstaticvoidmain(Stringargs[]){
for(inti=0;i<args.length;i++){
System.out.println("args ["+i+"]:"+args[i]);
}
}
}
如下所示,執行這個程式:
javaCommandLinethisisacommandline200-100
運行結果如下:
args[0]:this
args[1]:is
args[2]:a
args[3]:command
args[4]:line
args[5]:200
args[6]:-100
當一個物件被創建時候,構造方法被用來初始化該物件。構造方法和它所在類別的名字相同,但構造方法沒有回傳值。
通常會使用建構方法來賦予一個類別的實例變數賦初值,或執行其它必要的步驟來建立一個完整的物件。
不管你是否自訂構造方法,所有的類別都有構造方法,因為Java自動提供了一個預設構造方法,它將所有成員初始化為0。
一旦你定義了自己的構造方法,預設構造方法就會失效。
下面是一個使用構造方法的例子:
//一個簡單的建構子static classMyClass{
intx;
//以下是建構子MyClass(){
x=10;
}
}
你可以像下面這樣呼叫構造方法來初始化一個物件:
publicclassConsDemo{
publicstaticvoidmain(Stringargs[]){
MyClasst1=newMyClass();
MyClasst2=newMyClass();
System.out.println(t1.x+""+t2.x);
}
}
大多時候需要一個有參數的構造方法。
下面是一個使用構造方法的例子:
//一個簡單的建構子classMyClass{
intx;
//以下是建構子MyClass(inti){
x=i;
}
}
你可以像下面這樣呼叫構造方法來初始化一個物件:
publicclassConsDemo{
publicstaticvoidmain(Stringargs[]){
MyClasst1=newMyClass(10);
MyClasst2=newMyClass(20);
System.out.println(t1.x+""+t2.x);
}
}
運行結果如下:
1020
JDK 1.5 開始,Java支援將相同類型的可變參數傳遞給一個方法。
方法的可變參數的聲明如下所示:
typeName...parameterName
在方法宣告中,在指定參數類型後加一個省略號(...) 。
一個方法中只能指定一個可變參數,它必須是方法的最後一個參數。任何普通的參數必須在它之前聲明。
publicclassVarargsDemo{
publicstaticvoidmain(Stringargs[]){
//呼叫可變參數的方法 printMax(34,3,3,2,56.5);
printMax(newdouble[]{1,2,3});
}
publicstaticvoidprintMax(double...numbers){
if(numbers.length==0){
System.out.println("Noargumentpassed");
return;
}
doubleresult=numbers[0];
for(inti=1;i<numbers.length;i++)
if(numbers[i]>result){
result=numbers[i];
}
System.out.println("Themaxvalueis"+result);
}
}
以上實例編譯運行結果如下:
Themaxvalueis56.5
Themaxvalueis3.0
Java允許定義這樣的方法,它在物件被垃圾收集器析構(回收)之前調用,這個方法叫做finalize( ),它用來清除回收物件。
例如,你可以使用finalize()來確保一個物件開啟的檔案被關閉了。
在finalize()方法裡,你必須指定在物件銷毀時候要執行的操作。
finalize()一般格式是:
protectedvoidfinalize()
{
//在這裡終結程式碼}
關鍵字protected是一個限定符,它確保finalize() 方法不會被該類別以外的程式碼呼叫。
當然,Java的記憶體回收可以由JVM來自動完成。如果你手動使用,則可以使用上面的方法。
publicclassFinalizationDemo{
publicstaticvoidmain(String[]args){
Cakec1=newCake(1);
Cakec2=newCake(2);
Cakec3=newCake(3);
c2=c3=null;
System.gc();//呼叫Java垃圾收集器}
}
classCakeextendsObject{
privateintid;
publicCake(intid){
this.id=id;
System.out.println("CakeObject"+id+"iscreated");
}
protectedvoidfinalize()throwsjava.lang.Throwable{
super.finalize();
System.out.println("CakeObject"+id+"isdisposed");
}
}
運行以上程式碼,輸出結果如下:
C:1>javaFinalizationDemo
CakeObject1iscreated
CakeObject2iscreated
CakeObject3iscreated
CakeObject3isdisposed
CakeObject2isdisposed