Python類繫結方法及非繫結方法例項解析
一、繫結方法
1.物件的繫結方法
首先我們明確一個知識點,凡是類中的方法或函式,預設情況下都是繫結給物件使用的。下面,我們通過例項,來慢慢解析繫結方法的應用。
class People: def __init__(self,name,age): self.name = name self.age = age def talk(self): pass p = People('xiaohua',18) print(p.talk) 輸出結果: <bound method People.talk of <__main__.People object at 0x000000F802C69358>>
從上面的輸出結果來看,talk()這個類中的方法,是繫結給物件使用的。下面,我在看看另外一種情況。
class People: def __init__(self,age): self.name = name self.age = age def talk(): pass p = People('xiaohua',18) print(p.talk) 輸出結果: <bound method People.talk of <__main__.People object at 0x000000FF68F39358>>
現在,我們將talk()函式的引數去掉,結果顯示與上面是一樣。這說明,不管是類中的方法,還是類中函式,預設情況下都是繫結給物件使用的。繫結給物件使用有一種好處,那就是不用手動將物件傳入。物件是自動傳到類中。如果你不信,我們來看看下面的例子:
class People: def __init__(self,18) print(People.talk) print(p.talk) 輸出結果: <function People.talk at 0x000000C54E3D0A60> 類來呼叫僅僅是當作函式使用 <bound method People.talk of <__main__.People object at 0x000000C54E249358>> 而物件來呼叫則為繫結方法
上面很好說明了,如果類來呼叫類中的方法,那麼這個方法僅僅只是一個函式,那麼既然是函式,就不會有自動傳值這一功能。來看看下面程式碼:
class People: def __init__(self,age): self.name = name self.age = age def talk(self): pass p = People('xiaohua',18) People.talk() 1 p.talk() 2 #程式碼1處報錯 talk() missing 1 required positional argument: 'self' #程式碼2處正常
從上面輸出結果來看,當類呼叫類中的方法時候i,是不會進行自動傳值的,也就是說,函式有幾個引數,我們就得傳遞進去幾個引數。如果想結果正常執行,那麼在類名呼叫talk()的時候,將引數一一都傳遞進去。即:
People.talk(312312)
這個引數可以是任意的,但是,必須傳遞進去。而,當物件呼叫類中方法時候,則不用傳遞,如上面的2正常執行。既然知道了區別,那麼,我們來看看下面程式碼:
class People: def __init__(self,18) People.talk() 1 p.talk() 2 # 1處正常執行 # 2 處報錯 talk() takes 0 positional arguments but 1 was given
從輸出結果來看,People來呼叫talk()方法時候,並不需要傳遞引數;而當物件來呼叫talk()的時候,由於物件呼叫自己的繫結方法,會自動將物件當作第一個引數傳遞進去,所以,當類中talk()方法沒有帶引數時,而你又給它傳遞了一個,顯然是會報錯的。
綜上所述,我們可以得出以下結論:
1.凡是類中的方法和函式,都是繫結給物件使用的;
2.繫結方法都有自動傳值的功能。傳遞進去的值,就是物件本身。
3.如果類想呼叫繫結方法,就必須遵循函式的引數規則,有幾個引數,就必須傳遞幾個引數。
聰明的你,可能會問,既然類中的方法都是繫結給物件使用的,那麼有沒有方法是繫結給類使用的呢?
答案是,當然有!
2.類的繫結方法
既然類中的方法,預設都是繫結給物件使用,那麼,我們要採取一點措施,將類中的繫結方法解除物件繫結關係,進而繫結到類上。
在python中,引入了@classmethod方法,將類中的方法繫結到類身上。下面看看程式碼:
class People: @classmethod def talk(cls): pass p = People() print(People.talk) #輸出結果 <bound method People.talk of <class '__main__.People'>>
從上述結果可以看出,我們加上了一個裝飾器,將類中繫結給物件的方法,繫結到類身上了。我們之前分析過,如果一個方法繫結到誰身上,那麼在呼叫該函式的時候,將自動將該呼叫者當作第一個引數傳遞到函式中。但是,繫結到類的方法與繫結到物件方法有一點點不同:
class People: def __init__(self,name): self.name = name @classmethod def talk(cls): pass p = People('xiaohua') print(People.talk) print(p.talk) #輸出結果 <bound method People.talk of <class '__main__.People'>> <bound method People.talk of <class '__main__.People'>>
也就是說,當物件在呼叫類的繫結方法時,也會預設把類當作引數傳遞進去!所以下面執行正常,並不會因為這個方法繫結到類身上,而物件呼叫沒有傳遞引數,報錯!
class People: @classmethod def talk(cls): pass p = People() People.talk() p.talk()
但是,如果talk()沒有引數,則下面程式碼均會報錯。
class People: @classmethod def talk(): pass p = People() People.talk() p.talk() #報錯結果 talk() takes 0 positional arguments but 1 was given
兩者報錯結果一致,這就說明了,當物件來呼叫類的繫結方法時,也是自動將類傳遞進去,並不需遵循函式引數傳遞的規則。
對於類中的繫結方法,也基本上就這兩種,不管怎麼變化,只要記住以下規則,遇到這種情況,都不會再錯。
類中方法預設都是繫結給物件使用,當物件呼叫繫結方法時,會自動將物件作為第一個引數傳遞進去;而類來呼叫,則必須遵循函式引數一一對應的規則,有幾個引數,就必須傳遞幾個引數。如果一個方法是用了@classmethod裝飾器,那麼這個方法繫結到類身上,不管是物件來呼叫還是類呼叫,都會將類作為第一個引數傳遞進去。
二、非繫結方法
上面說了,類中的方法要麼是繫結給物件使用,要麼是繫結給類使用,那麼有沒有不繫結給兩者使用的函式?
答案:當然有,python給我們提供了@staticmethod,可以解除繫結關係,將一個類中的方法,變為一個普通函式。
下面,我們來看看程式碼示例:
import hashlib import time class MySQL: def __init__(self,host,port): self.id=self.create_id() self.host=host self.port=port @staticmethod def create_id(): #就是一個普通工具 m=hashlib.md5(str(time.clock()).encode('utf-8')) return m.hexdigest() print(MySQL.create_id) #<function MySQL.create_id at 0x0000000001E6B9D8> #檢視結果為普通函式 conn=MySQL('127.0.0.1',3306) print(conn.create_id) #<function MySQL.create_id at 0x00000000026FB9D8> #檢視結果為普通函式
從上面的輸出結果,我們可以看出,使用了@staticmethod裝飾了一個函式,那麼這個函式跟普通函式沒有什麼區別。既然是普通函式,那麼就遵從函式引數傳遞規則,有幾個引數就傳遞幾個引數。
以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支援我們。