編程小白 問關于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. 關于docker下的nginx壓力測試2. 為什么我ping不通我的docker容器呢???3. webpack - vuejs+java前后臺分離實現及部署問題4. mysql - 這條聯合sql語句哪里錯了5. 在html文件的目錄下輸入代碼按回車后顯示這個,哪位大佬幫幫我 呀6. linux pdo 安裝配置中 make 遇到問題,請大家指教!7. mysql - 我的myeclipse一直連顯示數據庫連接失敗,不知道為什么8. 漏洞 - 如何為MYSQL建立不使用的,偽裝的my.cnf文件?9. javascript - 微信h5發送圖文信息,部分設備點擊“發送”按鈕時沒反應,問題較難重現,如何能找到可能存在問題的點?10. 網站在移動的環境下手機,pc打不開
