編程小白 問(wèn)關(guān)于python當(dāng)中類的方法的參數(shù)問(wèn)題
問(wèn)題描述
代碼如下:
# -*- coding:gb2312 -*-class CarStore(object): def order(self,car_type):return Factory.select(car_type)class Factory(object): def select(car_type): #為什么這個(gè)地方的方法,沒(méi)有self參數(shù)也可以運(yùn)行?if car_type == '索納塔': return Suonata()elif car_type == '名圖': return Mingtu()else: print('沒(méi)有您要的車型!')class Car(object): def move(self):print('車在移動(dòng)...') def music(self):print('正在播放音樂(lè)...') 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()
執(zhí)行結(jié)果:
這段代碼本身是沒(méi)有問(wèn)題,可以執(zhí)行的。我的疑問(wèn)是,為什么第七行代碼def select(car_type): 這個(gè)地方明明沒(méi)有self,也可以執(zhí)行?我記得我在學(xué)類的時(shí)候,被教育實(shí)例方法里面每個(gè)方法都必須加上一個(gè)self的參數(shù),這里居然沒(méi)有,然后我在這個(gè)地方加上了self反而出錯(cuò)了,如下圖:
結(jié)果反而運(yùn)行出錯(cuò):
為什么會(huì)這樣?
問(wèn)題解答
回答1:你是不是用的python3?在python3中,如果成員函數(shù)不加self,則代表該函數(shù)是靜態(tài)成員函數(shù),可以直接使用“類名.函數(shù)名(參數(shù))”的形式直接調(diào)用。但是如果你加了self,則這個(gè)函數(shù)為類的成員函數(shù),在其他調(diào)用類中必須這樣調(diào)用:“類名.函數(shù)名(self,參數(shù))”,或者在調(diào)用類類中實(shí)現(xiàn)一個(gè)實(shí)例,“實(shí)例名.函數(shù)名(參數(shù))。
python3是上述這樣,我的測(cè)試,在python2中不是。
你在select函數(shù)中加了self,但是在 CarStore::order()中調(diào)用卻使用“類名.函數(shù)名(參數(shù))”的形式,所以不正確。你在CarStore::order()調(diào)用時(shí)改為“類名.函數(shù)名(self,參數(shù))”或者在CarStore::order()實(shí)現(xiàn)一個(gè)實(shí)例,使用“實(shí)例名.函數(shù)名(參數(shù))“這樣的形式試試。
昨晚看到的,太晚了,所以今天早上測(cè)試的。
回答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('沒(méi)有您要的車型!')
按這個(gè)寫法,select是對(duì)象方法,調(diào)用需要關(guān)聯(lián)一個(gè)實(shí)例Factory()。調(diào)用時(shí)對(duì)象實(shí)例是與第一個(gè)參數(shù)car_type綁定。這個(gè)參數(shù)名一般約定為self但是,不要求。
你需要明白以下兩種調(diào)用方式是不同的:
f = Factory()f.select(xxx)Factory.select(xxx)
第一種方式,用實(shí)例對(duì)象去調(diào)用,第一個(gè)參數(shù)car_type自動(dòng)與實(shí)例對(duì)象f綁定;第二種方式,用類去調(diào)用,第一個(gè)參數(shù)(car_type)沒(méi)有綁定;你需要自己綁定才能不出錯(cuò)——也就是你傳進(jìn)去的car_type:Factory.selct(car_type)這行。
但是,當(dāng)你加上self后,這個(gè)函數(shù)有兩個(gè)參數(shù),但是你只綁定了car_type,也就是綁定到第一個(gè)參數(shù)self,第二個(gè)沒(méi)有值,必然就出錯(cuò)了。
這里你要做的其實(shí)是將select實(shí)現(xiàn)成類方法:
class Factory(object): @classmethod def select(cls, car_type):if car_type == '索納塔': return Suonata()elif car_type == '名圖': return Mingtu()else: print('沒(méi)有您要的車型!')
那么,以Factory.select(car_type)調(diào)用時(shí),cls自動(dòng)綁定到Factory,而car_type則綁定到car_type。
以上,無(wú)論是self還是cls,都只是約定的名字而已,起作用的是Python的類-對(duì)象-方法模型。
建議可以看看《Python源碼剖析》,至少要理解@classmethod是怎么工作的,要不然是寫不好這類代碼的。
相關(guān)文章:
1. php多任務(wù)倒計(jì)時(shí)求助2. 數(shù)組排序,并把排序后的值存入到新數(shù)組中3. 默認(rèn)輸出類型為json,如何輸出html4. python中def定義的函數(shù)加括號(hào)和不加括號(hào)的區(qū)別?5. javascript - 有適合開(kāi)發(fā)手機(jī)端Html5網(wǎng)頁(yè)小游戲的前端框架嗎?6. PHP訂單派單系統(tǒng)7. python的正則怎么同時(shí)匹配兩個(gè)不同結(jié)果?8. javascript - jquery怎么讓a標(biāo)簽跳轉(zhuǎn)后保持tab的樣式9. mysql - sql 左連接結(jié)果union右連接結(jié)果,導(dǎo)致重復(fù)性計(jì)算怎么解決?10. javascript - charles map remote映射問(wèn)題
