Re: [問題] 關於class 內變數的問題
因為一樣是變數的問題,所以我直接回這個標題。
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
討論串 (同標題文章)
以下文章回應了本文:
完整討論串 (本文為第 5 之 7 篇):
Python 近期熱門文章
PTT數位生活區 即時熱門文章