Python 類可以定義專用方法,專用方法是在特殊情況下或當(dāng)使用特別語法時由 Python 替你調(diào)用的,而不是在代碼中直接調(diào)用(象普通的方法那樣)。
桂陽網(wǎng)站建設(shè)公司創(chuàng)新互聯(lián)建站,桂陽網(wǎng)站設(shè)計制作,有大型網(wǎng)站制作公司豐富經(jīng)驗。已為桂陽千余家提供企業(yè)網(wǎng)站建設(shè)服務(wù)。企業(yè)網(wǎng)站搭建\外貿(mào)營銷網(wǎng)站建設(shè)要多少錢,請找那個售后服務(wù)好的桂陽做網(wǎng)站的公司定做!
.1 __init__
類似于構(gòu)造函數(shù)
#!/usr/local/bin/python
class Study:
def __init__(self,name=None):
self.name = name
def say(self):
print self.name
study = Study("Badboy")
study.say()
.2 __del__
類似于析構(gòu)函數(shù)
#!/usr/local/bin/python
class Study:
def __init__(self,name=None):
self.name = name
def __del__(self):
print "Iamaway,baby!"
def say(self):
print self.name
study = Study("zhuzhengjun")
study.say()
.3__repr__
使用repr(obj)的時候,會自動調(diào)用__repr__函數(shù),該函數(shù)返回對象字符串表達式,
用于重建對象,如果eval(repr(obj))會得到一個對象的拷貝。
#!/usr/local/bin/python
class Study:
def __init__(self,name=None):
self.name = name
def __del__(self):
print "Iamaway,baby!"
def say(self):
print self.name
def __repr__(self):
return "Study('jacky')"
study = Study("zhuzhengjun")
study.say()
print type(repr(Study("zhuzhengjun"))) # str
print type(eval(repr(Study("zhuzhengjun")))) # instance
study = eval(repr(Study("zhuzhengjun")))
study.say()
.4__str__
Python能用print語句輸出內(nèi)建數(shù)據(jù)類型。有時,程序員希望定義一個類,要求它的對象也能用print語句輸出。Python類可定義特殊方法__str__,為類的對象提供一個不正式的字符串表示。如果類的客戶程序包含以下語句:
print objectOfClass
那么Python會調(diào)用對象的__str__方法,并輸出那個方法所返回的字符串。
#!/usr/local/bin/python
class PhoneNumber:
def __init__(self,number):
self.areaCode=number[1:4]
self.exchange=number[6:9]
self.line=number[10:14]
def __str__(self):
return "(%s) %s-%s"%(self.areaCode,self.exchange,self.line)
def test():
newNumber=raw_input("Enter phone number in the form. (123) 456-7890: \n")
phone=PhoneNumber(newNumber)
print "The phone number is:"
print phone
if__name__=="__main__":
test()
方法__init__接收一個形如"(xxx) xxx-xxxx"的字符串。字符串中的每個x都是電話號碼的一個位數(shù)。方法對字符串進行分解,并將電話號碼的不同部分作為屬性存儲。
方法__str__是一個特殊方法,它構(gòu)造并返回PhoneNumber類的一個對象的字符串表示。解析器一旦遇到如下語句:
print phone
就會執(zhí)行以下語句:
print phone.__str__()
程序如果將PhoneNumber對象傳給內(nèi)建函數(shù)str(如str(phone)),或者為PhoneNumber對象使用字符串格式化運算符%(例如"%s"%phone),Python也會調(diào)用__str__方法。
.5__cmp __
比較運算符,0:等于 1:大于 -1:小于
class Study:
def __cmp__(self, other):
if other 0 :
return 1
elif other 0:
return - 1
else:
return 0
study = Study()
if study -10:print 'ok1'
if study -10:print 'ok2'
if study == 0:print 'ok3'
打?。簅k2 ok3
說明:在對類進行比較時,python自動調(diào)用__cmp__方法,如-10 0 返回 -1,也就是說study 應(yīng)該小與 -10,估打印ok2
.6__getitem__
__getitem__ 專用方法很簡單。象普通的方法 clear,keys 和 values 一樣,它只是重定向到字典,返回字典的值。
class Zoo:
def __getitem__(self, key):
if key == 'dog':return 'dog'
elif key == 'pig':return 'pig'
elif key == 'wolf':return 'wolf'
else:return 'unknown'
zoo = Zoo()
print zoo['dog']
print zoo['pig']
print zoo['wolf']
打印 dog pig wolf
.7__setitem__
__setitem__ 簡單地重定向到真正的字典 self.data ,讓它來進行工作。
class Zoo:
def __setitem__(self, key, value):
print 'key=%s,value=%s' % (key, value)
zoo = Zoo()
zoo['a'] = 'a'
zoo['b'] = 'b'
zoo['c'] = 'c'
打?。?/p>
key=a,value=a
key=b,value=b
key=c,value=c
.8 __delitem__
__delitem__ 在調(diào)用 del instance[key] 時調(diào)用 ,你可能記得它作為從字典中刪除單個元素的方法。當(dāng)你在類實例中使用 del 時,Python 替你調(diào)用 __delitem__ 專用方法。
class A:
def __delitem__(self, key):
print 'delete item:%s' %key
a = A()
del a['key']
python中沒有函數(shù)重載。
為了考慮為什么python不提供函數(shù)重載,首先我們要研究為什么需要提供函數(shù)重載。
函數(shù)重載主要是為了解決兩個問題:
可變參數(shù)類型。
可變參數(shù)個數(shù)。
另外,一個基本的設(shè)計原則是,僅僅當(dāng)兩個函數(shù)除了參數(shù)類型和參數(shù)個數(shù)不同以外,其功能是完全相同的,此時才使用函數(shù)重載,如果兩個函數(shù)的功能其實不同,那么不應(yīng)當(dāng)使用重載,而應(yīng)當(dāng)使用一個名字不同的函數(shù)。
那么對于情況 1 ,函數(shù)功能相同,但是參數(shù)類型不同,python 如何處理?
答案是根本不需要處理,因為 python 可以接受任何類型的參數(shù),如果函數(shù)的功能相同,那么不同的參數(shù)類型在 python 中很可能是相同的代碼,沒有必要做成兩個不同函數(shù)。
那么對于情況 2 ,函數(shù)功能相同,但參數(shù)個數(shù)不同,python 如何處理?
答案就是缺省參數(shù)。對那些缺少的參數(shù)設(shè)定為缺省參數(shù)即可解決問題。因為你假設(shè)函數(shù)功能相同,那么那些缺少的參數(shù)終歸是需要用的。好了,鑒于情況 1 跟 情況 2 都有了解決方案,python 自然就不需要函數(shù)重載了。
更多Python知識請關(guān)注Python自學(xué)網(wǎng)
Python正確重載運算符的方法示例詳解
說到運算符重載相信大家都不陌生,運算符重載的作用是讓用戶定義的對象使用中綴運算符(如 + 和 |)或一元運算符(如 - 和 ~)。說得寬泛一些,在 Python 中,函數(shù)調(diào)用(())、屬性訪問(.)和元素訪問 / 切片([])也是運算符。
我們?yōu)?Vector 類簡略實現(xiàn)了幾個運算符。__add__ 和 __mul__ 方法是為了展示如何使用特殊方法重載運算符,不過有些小問題被我們忽視了。此外,我們定義的Vector2d.__eq__ 方法認(rèn)為 Vector(3, 4) == [3, 4] 是真的(True),這可能并不合理。下面來一起看看詳細(xì)的介紹吧。
運算符重載基礎(chǔ)
在某些圈子中,運算符重載的名聲并不好。這個語言特性可能(已經(jīng))被濫用,讓程序員困惑,導(dǎo)致缺陷和意料之外的性能瓶頸。但是,如果使用得當(dāng),API 會變得好用,代碼會變得易于閱讀。Python 施加了一些限制,做好了靈活性、可用性和安全性方面的平衡:
不能重載內(nèi)置類型的運算符
不能新建運算符,只能重載現(xiàn)有的
某些運算符不能重載——is、and、or 和 not(不過位運算符
、| 和 ~ 可以)
前面的博文已經(jīng)為 Vector 定義了一個中綴運算符,即 ==,這個運算符由__eq__ 方法支持。我們將改進 __eq__ 方法的實現(xiàn),更好地處理不是Vector 實例的操作數(shù)。然而,在運算符重載方面,眾多比較運算符(==、!=、、、=、=)是特例,因此我們首先將在 Vector 中重載四個算術(shù)運算符:一元運算符 - 和 +,以及中綴運算符 + 和 *。
一元運算符
-(__neg__)
一元取負(fù)算術(shù)運算符。如果 x 是 -2,那么 -x == 2。
+(__pos__)
一元取正算術(shù)運算符。通常,x == +x,但也有一些例外。如果好奇,請閱讀“x 和 +x 何時不相等”附注欄。
~(__invert__)
對整數(shù)按位取反,定義為 ~x == -(x+1)。如果 x 是 2,那么 ~x== -3。
支持一元運算符很簡單,只需實現(xiàn)相應(yīng)的特殊方法。這些特殊方法只有一個參數(shù),self。然后,使用符合所在類的邏輯實現(xiàn)。不過,要遵守運算符的一個基本規(guī)則:始終返回一個新對象。也就是說,不能修改self,要創(chuàng)建并返回合適類型的新實例。
對 - 和 + 來說,結(jié)果可能是與 self 同屬一類的實例。多數(shù)時候,+ 最好返回 self 的副本。abs(...) 的結(jié)果應(yīng)該是一個標(biāo)量。但是對 ~ 來說,很難說什么結(jié)果是合理的,因為可能不是處理整數(shù)的位,例如在ORM 中,SQL WHERE 子句應(yīng)該返回反集。
def __abs__(self):
return math.sqrt(sum(x * x for x in self))
def __neg__(self):
return Vector(-x for x in self) #為了計算 -v,構(gòu)建一個新 Vector 實例,把 self 的每個分量都取反
def __pos__(self):
return Vector(self) #為了計算 +v,構(gòu)建一個新 Vector 實例,傳入 self 的各個分量
x 和 +x 何時不相等
每個人都覺得 x == +x,而且在 Python 中,幾乎所有情況下都是這樣。但是,我在標(biāo)準(zhǔn)庫中找到兩例 x != +x 的情況。
第一例與 decimal.Decimal 類有關(guān)。如果 x 是 Decimal 實例,在算術(shù)運算的上下文中創(chuàng)建,然后在不同的上下文中計算 +x,那么 x!= +x。例如,x 所在的上下文使用某個精度,而計算 +x 時,精度變了,例如下面的
python里面沒有重載吧?
正常的重載說的應(yīng)該是函數(shù)方法的重載,即兩個函數(shù)的名稱相同但是參數(shù)不同是重載,會出現(xiàn)重載的情況一般是函數(shù)實現(xiàn)相同功能時,遇到參數(shù)類型可能不同的情況,會出現(xiàn)重載,如比較常見的對于加法的實現(xiàn),如果輸入兩個整數(shù)相加則直接使用整數(shù)加法,如果兩個字符串相加則使用字符串的拼接。輸入的參數(shù)類型不同,其實是兩個不同的函數(shù),這在那種強類型語言中是可以利用函數(shù)重載來實現(xiàn)的,而python中函數(shù)類型是動態(tài)的各種各樣的,所以不需要重載這個機制。相同函數(shù)名就是表示同一個方法。
沒有重載,但是可以有默認(rèn)參數(shù)和不定長參數(shù),可以判斷默認(rèn)值和參數(shù)長度來處理。
比如:
def range(start, end = -1):
if end == -1:
end = start
start = 0
或
def range(*args):
if len(args) == 1:
start = 0
end = args[0]
elif len(args) == 2:
start, end = args[0], args[1]
重載和重寫,這是兩個新概念,是兩個令我們?nèi)菀谆煜母拍?。方法重載(overloadingmethod)是在一個類里面,方法名字相同,而參數(shù)不同。返回類型呢?可以相同也可以不同。方法重寫(overidingmethod)子類不想原封不動地繼承父類的方法,而是想作一定的修改,這就需要采用方法的重寫。方法重寫又稱方法覆蓋。方法重載是讓類以統(tǒng)一的方式處理不同類型數(shù)據(jù)的一種手段。Java的方法重載,就是在類中可以創(chuàng)建多個方法,它們具有相同的名字,但具有不同的參數(shù)和不同的定義。調(diào)用方法時通過傳遞給它們的不同個數(shù)和類型的參數(shù)來決定具體使用哪個方法,這就是多態(tài)性。方法重寫:在Java中,子類可繼承父類中的方法,而不需要重新編寫相同的方法。但有時子類并不想原封不動地繼承父類的方法,而是想作一定的修改,這就需要采用方法的重寫。方法重寫又稱方法覆蓋。若子類中的方法與父類中的某一方法具有相同的方法名、返回類型和參數(shù)表,則新方法將覆蓋原有的方法。如需父類中原有的方法,可使用super關(guān)鍵字,該關(guān)鍵字引用了當(dāng)前類的父類重寫方法的規(guī)則:參數(shù)列表必須完全與被重寫的方法的相同,否則不能稱其為重寫而是重載.返回的類型必須一直與被重寫的方法的返回類型相同,否則不能稱其為重寫而是重載.訪問修飾符的限制一定要大于被重寫方法的訪問修飾符(publicprotecteddefaultprivate)重寫方法一定不能拋出新的檢查異?;蛘弑缺恢貙懛椒ㄉ昝鞲訉挿旱臋z查型異常.例如,父類的一個方法申明了一個檢查異常IOException,在重寫這個方法是就不能拋出Exception,只能拋出IOException的子類異常,可以拋出非檢查異常.重載的規(guī)則:必須具有不同的參數(shù)列表;可以有不同的返回類型,只要參數(shù)列表不同就可以了;可以有不同的訪問修飾符;可以拋出不同的異常;注意,Java的方法重載要求同名的方法必須有不同的參數(shù)表,僅有返回類型不同是不足以區(qū)分兩個重載的方法。重寫方法只能存在于具有繼承關(guān)系中,重寫方法只能重寫父類非私有的方法。下面分別舉一個例子來說明方法重載:publicclassTestOverLoad{publicstaticvoidmain(String[]args){Testtest=newTest();test.print(null);}}classTest{publicvoidprint(Stringsome){System.out.println("Stringversionprint");}publicvoidprint(Objectsome){System.out.println("Objectversionprint");}}該程序輸出的結(jié)果是Stringversionprint。