編程小白 問(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)有您要的車(chē)型!')class Car(object): def move(self):print('車(chē)在移動(dòng)...') def music(self):print('正在播放音樂(lè)...') def stop(self):print('車(chē)在停止...')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)有您要的車(chē)型!')
按這個(gè)寫(xiě)法,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)有您要的車(chē)型!')
那么,以Factory.select(car_type)調(diào)用時(shí),cls自動(dòng)綁定到Factory,而car_type則綁定到car_type。
以上,無(wú)論是self還是cls,都只是約定的名字而已,起作用的是Python的類-對(duì)象-方法模型。
建議可以看看《Python源碼剖析》,至少要理解@classmethod是怎么工作的,要不然是寫(xiě)不好這類代碼的。
相關(guān)文章:
1. javascript - 請(qǐng)問(wèn)為啥這個(gè)不能渲染成功?2. 老師您好!我有一個(gè)問(wèn)題、3. 后端開(kāi)發(fā) - mysql按時(shí)間分段統(tǒng)計(jì)的sql語(yǔ)句怎么寫(xiě)好?4. list - python 求助5. 哭遼 求大佬解答 控制器的join方法怎么轉(zhuǎn)模型方法6. mysql - 在下剛?cè)腴T(mén)sql 關(guān)于sql的語(yǔ)法詢問(wèn)7. javascript - iframe 為什么加載網(wǎng)頁(yè)的時(shí)候滾動(dòng)條這樣顯示?8. c++ - 如何在python的阻塞的函數(shù)中獲取變量值9. 初來(lái)乍到,相對(duì)路徑問(wèn)題,新手求教10. 網(wǎng)頁(yè)爬蟲(chóng) - python+requests 網(wǎng)頁(yè)重定向求解
