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

看板Python作者 (股海尋燈)時間16年前 (2009/01/22 04:46), 編輯推噓0(000)
留言0則, 0人參與, 最新討論串5/7 (看更多)
因為一樣是變數的問題,所以我直接回這個標題。 1、 a = 1 class testing: def test(self): a = 0 print a def test2(self): print a obj = testing() obj.test() obj.test2() print a ----------output 0 1 1 ----------- 看起來像是最上面第一行a的值並沒有被test()改成0,所以後面印出來的還是1。 2、 但是當我將a改成list的型態時,它似乎會被改變了。 a = [1] class testing: def test(self): a.append(2) print a def test2(self): print a obj = testing() obj.test() obj.test2() print a ---------output [1,2] [1,2] [1,2] --------- 看起來a在test()裡被改變了。 以上這2個例子讓我感到很困惑,一個會被改,一個不會被改, 想請問為什麼會有這樣子的差異呢? ※ 引述《sbrhsieh (sbr)》之銘言: : ※ 引述《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: 115.82.4.128
文章代碼(AID): #19TucaG6 (Python)
文章代碼(AID): #19TucaG6 (Python)