Re: [問題] 關於class 內變數的問題

看板Python作者 (sbr)時間16年前 (2009/01/22 02:22), 編輯推噓0(000)
留言0則, 0人參與, 最新討論串4/7 (看更多)
※ 引述《millerlai (Scent of Love)》之銘言: : 你的例子,我稍微改一下值。 : class CA: : V1 = 'V1' : def __init__(self): : self.V2 = 'V2' 以上這個 statement 執行後會產生一個 class object(應該說是 type instance) 這個 object bind 在 此 statement 所在 scope 中的名稱為 'CA'. 所以 CA.V1 可以說是 CA 所指涉的物件的 instance field. 這在 Java 中也是如此, static field 是 Class(derived) object 的 instance field. : 以下的程式碼,第一行建立了一個型別為CA的obj1 instance。 : 經過將obj1.V1設為"V_V" 之後,你覺得最後的輸出結果會是如何? : obj1 = CA(); 這個 statement 產生一個 object(CA instance), 該 object 有一個 instance field: 'V2'. : print(obj1.V1); : print(obj1.V2); : obj1.V1 = "V_V"; : print(CA.V1); : #---------output--------- : V1 : V2 : V1 : #---------output--------- : 在Java裡面,可透過 static 關鍵字來宣告一個 static field,但是在Python中 : static或instance field宣告無法從syntax上來判斷。 : 以上面的例子來說明,如果CA的V1欄位是一個static欄位,那麼最後一次print的結果 : 是V1卻不是V_V,這在Java的觀念裡是說不通的。 : Python是根據欄位的取用者來決定其欄位是static or instance。 : 當程式透過 obj1 取用V1欄位時,此時V1為instance欄位。 : 當程式透過 CA 取用V1欄位時,此時V1為static 欄位。 : 而CA在建構元中所寫的V2,是實體欄位沒有錯。 我也同意 CA.V1 不完全等同於 Java/C++ 中的 static field(data member) 這樣的 看法. 但是我認為某種程度上把 CA.V1 看成是一種 static field 也無不可. static field 在觀念上是屬於 Class object(此 Class 是以 Java 界的說法), 也就是說 static field 是保存(依附)在 Class object 本身(不是依附在該 Class 建構出來的 instance 上. 在 Python 中也是如此, CA.V1 是依附在 CA 所指涉的 type instance 中, 由 CA 所建構的多個 instance 共享. Python 中的 object(a type name) 所實作的 __getattribute__ method, 除了會在 物件本身的 namespace 裡尋找外, 也會蒐尋物件所屬的 class object, 所以上例中 在尚未 assign 一個物件進 obj1 的 namespace 之前, obj1.V1 也可以存取到 CA.V1 所綁定的值. obj1.V1 = "V_V" 這個 statement 是把 "V_V" 這個 object bind 在 obj1(指涉的物件)的 namespace(或說 __dict__), 而遮蔽了 CA.V1. 以下這個 Java code 可以說跟文中前段的 Python code 有很高程度上的相似性. public class CA { public static String V1 = "V1"; public String V2; public String _V1; public CA() { this.V2 = "V2"; } public static void main(String[] args) { CA obj1 = new CA(); System.out.println(obj1.getV1()); System.out.println(obj1.V2); obj1.setV1("V_V"); System.out.println(CA.V1); } public String getV1() { return _V1 != null? _V1 : V1; } public void setV1(String value) { this._V1 = value; } } 基於 Java 中的 class member(field/method)在 class resolve 之後就固定, 所以 只能用依據一個 field 的值是否為空(null)來模擬 field 是否存在. -- ※ 發信站: 批踢踢實業坊(ptt.cc) ◆ From: 218.173.129.48 ※ 編輯: sbrhsieh 來自: 218.173.129.48 (01/22 02:36)
文章代碼(AID): #19TsV-YW (Python)
文章代碼(AID): #19TsV-YW (Python)