編程小白 問關于python當中類的方法的參數問題
問題描述
代碼如下:
# -*- coding:gb2312 -*-class CarStore(object): def order(self,car_type):return Factory.select(car_type)class Factory(object): def select(car_type): #為什么這個地方的方法,沒有self參數也可以運行?if car_type == '索納塔': return Suonata()elif car_type == '名圖': return Mingtu()else: print('沒有您要的車型!')class Car(object): def move(self):print('車在移動...') def music(self):print('正在播放音樂...') def stop(self):print('車在停止...')class Suonata(Car): def __init__(self):print('索納塔')class Mingtu(Car): def __init__(self):print('名圖')car_store = CarStore()suonata = car_store.order('索納塔')#car.move()#car.music()#car.stop()suonata.move()
執行結果:
這段代碼本身是沒有問題,可以執行的。我的疑問是,為什么第七行代碼def select(car_type): 這個地方明明沒有self,也可以執行?我記得我在學類的時候,被教育實例方法里面每個方法都必須加上一個self的參數,這里居然沒有,然后我在這個地方加上了self反而出錯了,如下圖:
結果反而運行出錯:
為什么會這樣?
問題解答
回答1:你是不是用的python3?在python3中,如果成員函數不加self,則代表該函數是靜態成員函數,可以直接使用“類名.函數名(參數)”的形式直接調用。但是如果你加了self,則這個函數為類的成員函數,在其他調用類中必須這樣調用:“類名.函數名(self,參數)”,或者在調用類類中實現一個實例,“實例名.函數名(參數)。
python3是上述這樣,我的測試,在python2中不是。
你在select函數中加了self,但是在 CarStore::order()中調用卻使用“類名.函數名(參數)”的形式,所以不正確。你在CarStore::order()調用時改為“類名.函數名(self,參數)”或者在CarStore::order()實現一個實例,使用“實例名.函數名(參數)“這樣的形式試試。
昨晚看到的,太晚了,所以今天早上測試的。
回答2:self 不是魔法,改成什么名字都可以的
不一定非要 self
回答3:4 return Factory.select(self, car_type)回答4:
class Factory(object): def select(car_type):if car_type == '索納塔': return Suonata()elif car_type == '名圖': return Mingtu()else: print('沒有您要的車型!')
按這個寫法,select是對象方法,調用需要關聯一個實例Factory()。調用時對象實例是與第一個參數car_type綁定。這個參數名一般約定為self但是,不要求。
你需要明白以下兩種調用方式是不同的:
f = Factory()f.select(xxx)Factory.select(xxx)
第一種方式,用實例對象去調用,第一個參數car_type自動與實例對象f綁定;第二種方式,用類去調用,第一個參數(car_type)沒有綁定;你需要自己綁定才能不出錯——也就是你傳進去的car_type:Factory.selct(car_type)這行。
但是,當你加上self后,這個函數有兩個參數,但是你只綁定了car_type,也就是綁定到第一個參數self,第二個沒有值,必然就出錯了。
這里你要做的其實是將select實現成類方法:
class Factory(object): @classmethod def select(cls, car_type):if car_type == '索納塔': return Suonata()elif car_type == '名圖': return Mingtu()else: print('沒有您要的車型!')
那么,以Factory.select(car_type)調用時,cls自動綁定到Factory,而car_type則綁定到car_type。
以上,無論是self還是cls,都只是約定的名字而已,起作用的是Python的類-對象-方法模型。
建議可以看看《Python源碼剖析》,至少要理解@classmethod是怎么工作的,要不然是寫不好這類代碼的。
相關文章:
1. angular.js - angularjs 使用鼠標懸停時,標簽一直閃2. 一個走錯路的23歲傻小子的提問3. c++ - win764位環境下,我用GCC為什么指針占8個字節,而long是4個字節?4. html5 - HTML代碼中的文字亂碼是怎么回事?5. node.js - 函數getByName()中如何使得co執行完后才return6. python - django 里自定義的 login 方法,如何使用 login_required()7. android - 安卓activity無法填充屏幕8. python 計算兩個時間相差的分鐘數,超過一天時計算不對9. javascript - SuperSlide.js火狐不兼容怎么回事呢10. java - 安卓電視盒子取得了root權限但是不能安裝第三方應用,請問該怎么辦?
