在Java程式設計中經常會見到this的使用,this使得程式設計變得規範、簡單、靈活。但是在使用過程中,在不同場 合它的含義並不完全相同,使用不當還會出現錯誤, 本文對this的幾種用法和出現的問題進行了分析詳解。 關鍵詞:類;對象;this;成員變數;方法;構造方法 中,Java語言提供了豐富的 ...
在Java程式設計中經常會見到this的使用,this使得程式設計變得規範、簡單、靈活。但是在使用過程中,在不同場
合它的含義並不完全相同,使用不當還會出現錯誤,
本文對this的幾種用法和出現的問題進行了分析詳解。
關鍵詞:類;對象;this;成員變數;方法;構造方法
中,Java語言提供了豐富的類(Class)、介面(Interface)以及相應的方法(Method)。
使用這些類或介面,用戶可以定義
自己的類或子類,並以這些類為模板創建對象(Object)。
在Java語言中,當創建一個對象後,Java虛擬機就會為其分配一個指向對象本身的指針,這個指針就是“this”。關鍵字this與對象密切相
關,在Java程式設計中經常會見到this的使用,我們很多人很難理解它,因為它的語法較為靈活,那this到底有哪些用處呢?什麼情況下使用它呢?本文就具體來分析一下this的使用方法。
一、使用this調用本類中的成員變數(屬性)
例1:觀察以下的程式代碼,看看會出現哪些問題呢?
class Student { private String name; //定義一個成員變數name public void setName(String name) {//設置姓名,定義一個形參name name=name; //將局部變數的值傳遞給成員變數 } public String getName( ) {//獲得姓名 return "姓名:"+name; }} public class ThisExample01 { public static void main(String args[]) { Student stu=new Student(); stu.setName("李明"); System.out.println(stu.getName()); }}
運行結果:姓名:null
由運行結果可以看出,通過name=name並沒有正確的將內容賦給屬性,為什麼呢?因為此時操作的name實際上是方法中的,跟類中的屬性完全不沾邊。
另外,這個程式代碼中,有一個成員變數name,同時在方法中有個形式參數,參數名也是name。然後在方法中將形式參數name的值傳遞給成員變數name。雖然我們可以看明白這個代碼的含義,但是作
為Java編譯器它是怎麼判斷使用哪個變數的呢?到底是將形式參數name的值傳遞給成員變數name,還是反過來講成員變數name的值傳遞給形式參數name呢?
此時this這個關鍵字就起到作用了,這種情
況下使用this這個關鍵字代表的就是類中的成員變數,又叫做類的屬性。所以此時,為了明確的表示出哪一個是類中的屬性,就要加上“this.屬性名稱”的操作,
將student類的代碼改進如下:
class Student { private String name; public void setName(String name) { http://www.wenkuxiazai.com=name; //將形 參的值傳遞給成員變數 } public String getName( ) { return "姓名:"+name; }}
這時,代表的就是類中的成員變數,而賦值號右邊的name則是方法的形式參數,代碼name就是將形
式參數的值傳遞給成員變數。
例1只是以一個形式參數為例。其實如果是局部變數的話,也是相同的道理。在方法體內定義的變數稱為局部變數,在類的內部方法體的外部定義的變數稱為成員變數。如果成員變數和方法中的局部變
量的名字相同時,那麼在方法中成員變數將會被屏蔽。如果這個時候還要使用成員變數,就需要使用關鍵字this。使用this引用成員變數的方法格式:this.成員變數名。
既然this能夠調用本類中的成員變數,那麼,this也可以調用本類中的成員方法。以例2為例,程式代碼如下:
class Student { private String name; public void setName(String name) { this.print();//調用本類中的print方法 } public String getName( ) { return "姓名:"+name; } public void print( ) { System.out.println("設置相關信息如下……"); }} public class ThisExample02 { public static void main(String args[]) { Student stu=new Student(); stu.setName("李明"); System.out.println(stu.getName()); }}
運行結果:
設置相關信息如下…… 李明
一般情況下,在Java語言中引用成員變數或者成員方法都是以對象名.成員變數或者對象名.成員方法的形式。不過有些程式員即使在沒有相同變數的時候,也喜歡使用this.成員變數的形式來引用變數
。這主要是從便於代碼的閱讀考慮。一看到這個this關鍵字就知道現在引用的變數是成員變數或者成員方法,而不是局部變數。這無形中就提高了代碼的閱讀性。
二、使用this調用構造方法
在一個Java類中,構造方法是一個與類同名的方法,必須與類的名字相同。而且在Java類中必須存在一個構造方法。如果在代碼中沒有顯示的體現構造方法的話,那麼編譯器在編譯的時候會自動添加
一個沒有形式參數的構造方法。在一個類中可以存在多個構造方法,這些構造方法都採用相同的名字,只是形式參數不同。Java語言就憑用戶的參數來判斷調用哪一個構造方法。當一個類中有多個構
造方法時,可以利用this關鍵字相互調用。假設,現在有一個類中存在多個構造方法,但是不管有多少個構造方法,只要對象一被實例化,就必須列印一句“新對象實例化”的信息出來,這時可以有
兩種做法。
第一種做法:按照最原始的方式完成,具體代碼如下:
class Student
其他方法引用構造方法一樣,都是通過形式參數來調用構造方
法。Java編譯器會根據所傳遞的參數數量的不同,來判斷該調用哪個構造方法。
所以,我們在實際編程的時候有時候需要在一個構造方法中對另一個構造方法進行調用。但是,在使用this關鍵字調用其他構造方法的時候,this()調用構造方法只能放在構造方法的首行,為的是能
夠為類中的屬性初始化;而且至少有一個構造方法是不用this調用,否則程式會出現錯誤。
class Student { private String name ; private int age ; public Student () { this("李明",20) ;//調用有兩個參數的構造方法 System.out.println("新對象實例化") ; } public Student (String name) { this() ; } public Student (String name,int age) { this(name) ; this.age = age ; } public String getInfo(){ return "姓名:" + name + ",年齡:" + age ; }} public class ThisExample05 { public static void main(String args[]) { Student stu1 = new Student ("李小明",19) ; System.out.println(stu1.getInfo()) ; }} 這時候構造方法就出現了遞歸調用,程式出錯。
註意的是,使用this調用構造方法只適用於構造方法的調用,類中的其他方法不能使用這種方法。
三、使用this引用當前對象
this最重要的特定就是表示當前對象,那什麼叫當前對象呢?在Java中當前對象就是指當前正在調用類中方法的對象。使用this引用當前對象是指如果在類的方法中需要返回一個對象,並且該對象是
方法所在的類的當前對象,可以使用this關鍵字作為方法的返回值。例如:
class Student { public String getInfo()//取得信息的方法 { System.out.println("Student類 --> " + this) ; // 直接列印this return null ; //為了保證語法正確,返回null }} public class ThisExample06{ public static void main(String args[]) { Student stu1 = new Student();//調用構造實例化對象 Student stu2 = new Student();//調用構造實例化對象 System.out.println("MAIN方法 --> " +stu1) ; //直接列印對象 stu1.getInfo() ; // 當前調用getInfo()方法的對象是stu1 System.out.println("MAIN方法 --> " +stu2) ; //直接列印對象 { private String name //姓名 private int age ; //年齡 public Student (){// 無參構造System.out.println("新對象實例化") ; } public Student (String name) { System.out.println("新對象實例化") ; } public Student (String name,int age) // 通過構造方法賦值 { System.out.println("新對象實例化") ; //為類中的name屬性賦值 this.age = age ; //為類中的age屬性賦值 } public String getInfo(){//取得信息的方法return "姓名:" + name + ",年齡:" + age ; }} public class ThisExample03 { public static void main(String args[]) { Student stu1 = new Student ("李明",20) ; //調用構造實例化對象 System.out.println(stu1.getInfo()) ; //取得信息 }}
這個例子中,一個無參的構造方法,一個提供一個參數用於設置姓名的構造方法,還有一個提供兩個參數用於設置姓名和年齡的構造方法,這三個方法都是用來列印新對象實例化的信息,很明顯,此
時如果在各個構造方法中編寫輸出語句肯定是不合適的,其中有一些代碼重覆了,現在只是一行,所以感覺不出來,如果現在的代碼有很多行的話,以上代碼的缺陷就立刻顯現出來了。那麼,最好讓
構造方法間進行相互的調用,這時就可以用“this(參數列表)”的形式完成,用this修改以上的代碼如下:
class Student { private String name ; private int age ; public Student(){ System.out.println("新對象實例化") ; } public Student (String name) { this() ; //調用本類中的無參構造方法 http://www.wenkuxiazai.com = name ; } public Student (String name,int age) { this(name); //調用有一個參數的構造方法 this.age = age ; } public String getInfo(){ //取得信息的方 法 return "姓名:" + name + ",年齡:" + age ; }} public class ThisExample04 { public static void main(String args[]) { Student stu1 = new Student ("李明",20) ; System.out.println(stu1.getInfo()) ; } }
一個類中有多個構造方法,因為其名字都相同,跟類名一致,那麼這個this到底是調用哪個構造方法呢?
stu2.getInfo() ; // 當前調用getInfo()方法
}}
我們再用一個例子來解釋this引用當前對象。public class Car
{ public Car getCarObject(){ return this; //返回當前對象}
public static void main(String[] args) { Car sc = new Car ();//創建一個Car對象 System.out.println( sc.getCarObject() instanceof Car);
}}
這裡定義了一個返回類型為Car類型的方法getCarObject(),並使用this關鍵字返回當前的對象Car。在main()方法中創建一個Car對象並使用instanceof方法判斷getCarObject()方法返回的對象與Car
對象是否匹配。運行結果為true。
四、其他用法
除了上述情況外,this還可以用在其他場合。例如對象的比較,可以使用this和引用傳遞進行兩個對象是否相等的判斷。例如,一個學生類如下:
class Student { private String name ; private int age ; public Student (String name,int age) { this.setName(name) ; this.setAge(age) ; } public void setName(String name){ //設置姓名 } public void setAge(int age){ //設置年齡 this.age = age ; } public String getName(){ return; } public int getAge(){ return this.age ; }}
生成兩個對象,當兩個對象中的姓名和年齡完全相等的時候,則認為對象相等,但是在此時就產生了兩個問題:一是如何進行對象的比較?二是在哪裡進行比較?這個例子中姓名定義為String類型,
我們知道String本身事一個類,如果想要進行相等的比較,則要判斷內容是否相等,必須使用equals()方法完成。而年齡age定義為整型,直接使用“==”就可以完成。另外,對象的比較應該由自己進
行比較最為合適,所以應該在Student類中增加一個比較的方法,具體代碼如下:
class Student { private String name ; private int age ; public Student (String name,int age) { this.setName(name) ;this.setAge(age) ; } public boolean compare(Student stu) { //調用此方法時裡面存在兩個對象:當前對象、傳入的對象 Student s1 = this ;//當前的對象,就表示stu1 Student s2 = stu ;//傳遞進來的對象,就表示stu2 if(s1==s2){// 判斷是不是同一個對象,用地址比較 return true ; } //之後分別判斷每一個屬性是否相等 if(equals()&&s1.age==s2.age){ return true ; //兩個對象相等 }else{ return false ; //兩個對象不相等 } } public void setName(String name){ //設置姓名 ; } public void setAge (int age){ //設置年齡 this.age = age ; } public String getName(){ return; } public int getAge(){ return this.age ; }} public class ThisExample07 { public static void main(String args[]) { Student stu1 = new Student ("李明",20) ; //聲明兩個對象,內容完全相等 Student stu2 = new Student ("李明",20) ; //聲明兩個對象,內容完全相等 // 直接在主方法中依次取得各個屬性進行比較 if(stu1.compare(stu2)){ System.out.println("兩個對象相等!") ; }else{ System.out.println("兩個對象不相等!") ; } }}
地址相等,則兩個對象相等;如果地址不相等,則依次判斷每一個屬性的內容來判斷其是否相等。
結束語:Java中的this與面向對象的程式設計密切相關,它表示當前對象。
但是,在不同的使用場合它表示的真正含義是不完全一樣的。this.成員變數、this.成員方法(參數列表):實際上都表示當
前對象中的屬性或當前對象調用的方法;this的核心表示當前對象,當前正在操作本方法的對象稱為當前對象;使用this可以調用其他構造方法,但是此語句必須放在構造方法的首行
this的使用在 節省代碼、區分局部變數與成員變數、對象的比較等方面起著重要的作用,編程時我們應當給予重視並能夠學會正確使用它。