溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點(diǎn)擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

python面向?qū)ο蠓椒ǖ膮^(qū)別是什么

發(fā)布時(shí)間:2020-09-09 14:08:18 來源:億速云 閱讀:139 作者:小新 欄目:編程語言

這篇文章主要介紹python面向?qū)ο蠓椒ǖ膮^(qū)別是什么,文中介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們一定要看完!

                                                           Python從設(shè)計(jì)之初就已經(jīng)是一門面向?qū)ο蟮恼Z言,正因?yàn)槿绱?,在Python中創(chuàng)建一個(gè)類和對象是很容易的。

方法包括:實(shí)例方法、靜態(tài)方法和類方法,三種方法在內(nèi)存中都?xì)w屬于類,區(qū)別在于調(diào)用方式不同。

實(shí)例方法:由對象調(diào)用;至少一個(gè)self參數(shù);執(zhí)行普通方法時(shí),自動將調(diào)用該方法的對象賦值給self;

類方法:由類調(diào)用; 至少一個(gè)cls參數(shù);執(zhí)行類方法時(shí),自動將調(diào)用該方法的類復(fù)制給cls;

靜態(tài)方法:由類調(diào)用;無默認(rèn)參數(shù)。

python面向?qū)ο蠓椒ǖ膮^(qū)別是什么

實(shí)例方法

class Kls(object):
    def __init__(self, data):
        self.data = data
    def printd(self):
        print(self.data)
ik1 = Kls('leo')
ik2 = Kls('lee')
ik1.printd()
ik2.printd()

輸出:

leo 
lee

上述例子中,printd為一個(gè)實(shí)例方法。實(shí)例方法第一個(gè)參數(shù)為self,當(dāng)使用ik1.printd()調(diào)用實(shí)例方法時(shí),實(shí)例ik1會傳遞給self參數(shù),這樣self參數(shù)就可以引用當(dāng)前正在調(diào)用實(shí)例方法的實(shí)例。利用實(shí)例方法的這個(gè)特性,上述代碼正確輸出了兩個(gè)實(shí)例的成員數(shù)據(jù)。

類方法

Python 的類方法采用裝飾器@classmethod來定義,我們直接看例子。

class Kls(object):
    num_inst = 0
    def __init__(self):
        Kls.num_inst = Kls.num_inst + 1
    @classmethod
    def get_no_of_instance(cls):
        return cls.num_inst
ik1 = Kls()
ik2 = Kls()
print ik1.get_no_of_instance()
print Kls.get_no_of_instance()

輸出:

2 
2

在上述例子中,我們需要統(tǒng)計(jì)類Kls實(shí)例的個(gè)數(shù),因此定義了一個(gè)類變量num_inst來存放實(shí)例個(gè)數(shù)。通過裝飾器@classmethod的使用,方法get_no_of_instance被定義成一個(gè)類方法。在調(diào)用類方法時(shí),Python 會將類(class Kls)傳遞給cls,這樣在get_no_of_instance內(nèi)部就可以引用類變量num_inst。

由于在調(diào)用類方法時(shí),只需要將類型本身傳遞給類方法,因此,既可以通過類也可以通過實(shí)例來調(diào)用類方法。

靜態(tài)方法

在開發(fā)中,我們常常需要定義一些方法,這些方法跟類有關(guān),但在實(shí)現(xiàn)時(shí)并不需要引用類或者實(shí)例,例如,設(shè)置環(huán)境變量,修改另一個(gè)類的變量,等。這個(gè)時(shí)候,我們可以使用靜態(tài)方法。

Python 使用裝飾器@staticmethod來定義一個(gè)靜態(tài)方法。

IND = 'ON'

class Kls(object):
    def __init__(self, data):
        self.data = data
    @staticmethod
    def checkind():
        return IND == 'ON'
    def do_reset(self):
        if self.checkind():
            print('Reset done for: %s' % self.data)
    def set_db(self):
        if self.checkind():
            print('DB connection made for: %s' % self.data)
ik1 = Kls(24)
ik1.do_reset()
ik1.set_db()

輸出:

Reset done for: 24 
DB connection made for: 24

在代碼中,我們定義了一個(gè)全局變量IND,由于IND跟類Kls相關(guān),所以我們將方法checkind放置在類Kls中定義。方法checkind只需檢查IND的值,而不需要引用類或者實(shí)例,因此,我們將方法checkind定義為靜態(tài)方法。

對于靜態(tài)方法,Python 并不需要傳遞類或者實(shí)例,因此,既可以使用類也可以使用實(shí)例來調(diào)用靜態(tài)方法。

實(shí)例方法,類方法與靜態(tài)方法的區(qū)別

我們用代碼說明實(shí)例方法,類方法,靜態(tài)方法的區(qū)別。注意下述代碼中方法foo,class_foo,static_foo的定義以及使用。

class Kls(object):
    def foo(self, x):
        print('executing foo(%s,%s)' % (self, x))
    @classmethod
    def class_foo(cls,x):
        print('executing class_foo(%s,%s)' % (cls,x))
    @staticmethod
    def static_foo(x):
        print('executing static_foo(%s)' % x)
ik = Kls()
# 實(shí)例方法
ik.foo(1)
print(ik.foo)
print('==========================================')
# 類方法
ik.class_foo(1)
Kls.class_foo(1)
print(ik.class_foo)
print('==========================================')
# 靜態(tài)方法
ik.static_foo(1)
Kls.static_foo('hi')
print(ik.static_foo)

輸出:

executing foo(<__main__.Kls object at 0x0551E190>,1)
<bound method Kls.foo of <__main__.Kls object at 0x0551E190>>
==========================================
executing class_foo(<class '__main__.Kls'>,1)
executing class_foo(<class '__main__.Kls'>,1)
<bound method type.class_foo of <class '__main__.Kls'>>
==========================================
executing static_foo(1)
executing static_foo(hi)
<function static_foo at 0x055238B0>

對于實(shí)例方法,調(diào)用時(shí)會把實(shí)例ik作為第一個(gè)參數(shù)傳遞給self參數(shù)。因此,調(diào)用ik.foo(1)時(shí)輸出了實(shí)例ik的地址。

對于類方法,調(diào)用時(shí)會把類Kls作為第一個(gè)參數(shù)傳遞給cls參數(shù)。因此,調(diào)用ik.class_foo(1)時(shí)輸出了Kls類型信息。

前面提到,可以通過類也可以通過實(shí)例來調(diào)用類方法,在上述代碼中,我們再一次進(jìn)行了驗(yàn)證。

對于靜態(tài)方法,調(diào)用時(shí)并不需要傳遞類或者實(shí)例。其實(shí),靜態(tài)方法很像我們在類外定義的函數(shù),只不過靜態(tài)方法可以通過類或者實(shí)例來調(diào)用而已。

值得注意的是,在上述例子中,foo只是個(gè)函數(shù),但當(dāng)調(diào)用ik.foo的時(shí)候我們得到的是一個(gè)已經(jīng)跟實(shí)例ik綁定的函數(shù)。調(diào)用foo時(shí)需要兩個(gè)參數(shù),但調(diào)用ik.foo時(shí)只需要一個(gè)參數(shù)。foo跟ik進(jìn)行了綁定,因此,當(dāng)我們打印ik.foo時(shí),會看到以下輸出:

<bound method Kls.foo of <__main__.Kls object at 0x0551E190>>

當(dāng)調(diào)用ik.class_foo時(shí),由于class_foo是類方法,因此,class_foo跟Kls進(jìn)行了綁定(而不是跟ik綁定)。當(dāng)我們打印ik.class_foo時(shí),輸出:

<bound method type.class_foo of <class '__main__.Kls'>>

當(dāng)調(diào)用ik.static_foo時(shí),靜態(tài)方法并不會與類或者實(shí)例綁定,因此,打印ik.static_foo(或者Kls.static_foo)時(shí)輸出:

<function static_foo at 0x055238B0>

概括來說,是否與類或者實(shí)例進(jìn)行綁定,這就是實(shí)例方法,類方法,靜態(tài)方法的區(qū)別。

以上是python面向?qū)ο蠓椒ǖ膮^(qū)別是什么的所有內(nèi)容,感謝各位的閱讀!希望分享的內(nèi)容對大家有幫助,更多相關(guān)知識,歡迎關(guān)注億速云行業(yè)資訊頻道!

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI