小蛇學(xué)python(19)裝飾器

python的裝飾器是python的特色高級(jí)功能之一,言簡意賅得說,其作用是在不改變其原有函數(shù)和類的定義的基礎(chǔ)上,給他們增添新的功能。

裝飾器存在的意義是什么呢?我們知道,在python中函數(shù)可以調(diào)用,類可以繼承,為何要必須保證不改變函數(shù)和類的定義,就使得函數(shù)有了新的功能呢?其實(shí)很好解釋。

提高代碼的簡潔程度與封裝性。如果你采用新聲明一個(gè)函數(shù),并調(diào)用原來函數(shù)的思路使得原函數(shù)功能增加了,但是一方面使用起來看著不簡潔, 另一方面當(dāng)另一個(gè)程序員使用你的代碼時(shí)再使用這樣的思路,那代碼嵌套無窮無盡,會(huì)讓代碼變得很亂,說不定改錯(cuò)了哪里,動(dòng)了原函數(shù)哪個(gè)參數(shù),就會(huì)出現(xiàn)發(fā)現(xiàn)不了的bug。

我們先來看一個(gè)簡單的例子。

def addworld(func):
    def addfun():
        return func() + 'world'
    return addfun()

@addworld
def printhello():
    return 'hello'

print(printhello)

這段代碼最后運(yùn)行出來的結(jié)果是helloworld。我們發(fā)現(xiàn),只要在原函數(shù)頭上加一個(gè)你定義好了的增添功能的模板,以后凡是加上這個(gè),都會(huì)增添一個(gè)功能,這提高了開發(fā)效率,也看著更加簡潔。

def addnum(func):
    def addfun(*args, **kwargs):
        ret = func(*args, **kwargs)*2
        return ret
    return addfun

@addnum
def printsum(a, b):
    return (a+b)

i = printsum(7, 7)
print(i)

這是有參數(shù)的情況下,裝飾器的使用效果,輸出結(jié)果為28。

def addnum1(c):
    def addnum(func):
        def addfun(*args, **kwargs):
            ret = func(*args, **kwargs)*2*c
            return ret
        return addfun
    return addnum

@addnum1(5)
def printsum(a, b):
    return (a+b)

i = printsum(7, 7)
print(i)

裝飾器本身也是可以傳參的,在這個(gè)程序中,傳入了一個(gè)常數(shù)5,最后得出的結(jié)果是140。

接下來,我們介紹幾個(gè)常用的python內(nèi)置裝飾器。

property

在綁定屬性時(shí),如果我們直接把屬性暴露出去,雖然寫起來很簡單,但是,沒辦法檢查參數(shù),導(dǎo)致可以把成績隨便改。這樣顯然是不行的,那么在函數(shù)中,我們往往如何提高程序的魯棒性呢?

class Student(object):

    def get_score(self):
        return self._score

    def set_score(self, value):
        if not isinstance(value, int):
            raise ValueError('score must be an integer!')
        if value < 0 or value > 100:
            raise ValueError('score must between 0 ~ 100!')
        self._score = value

s = Student()
s.set_score(60)
value = s.get_score()
print(value)

通過調(diào)用類中的函數(shù)從而進(jìn)一步操作屬性,這樣雖然安全正確卻未免看著復(fù)雜,不貼近人類自然的屬性表現(xiàn)。

class Student(object):

    @property
    def score(self):
        return self._score

    @score.setter
    def score(self, value):
        if not isinstance(value, int):
            raise ValueError('score must be an integer!')
        if value < 0 or value > 100:
            raise ValueError('score must between 0 ~ 100!')
        self._score = value

s = Student()
s.score = 80
print(s.score)

而加上一個(gè)@property就可以輕松解決這個(gè)問題,是不是很奇妙?把一個(gè)getter方法變成屬性,只需要加上@property就可以了,此時(shí),@property本身又創(chuàng)建了另一個(gè)裝飾器@score.setter,負(fù)責(zé)把一個(gè)setter方法變成屬性賦值,于是,我們就擁有一個(gè)可控的屬性操作。

還可以定義只讀屬性,只定義getter方法,不定義setter方法就是一個(gè)只讀屬性。

class Student(object):

    @property
    def birth(self):
        return self._birth

    @birth.setter
    def birth(self, value):
        self._birth = value

    @property
    def age(self):
        return 2014 - self._birth

上面的birth是可讀寫屬性,而age就是一個(gè)只讀屬性,因?yàn)閍ge可以根據(jù)birth和當(dāng)前時(shí)間計(jì)算出來。

我的博客即將搬運(yùn)同步至騰訊云+社區(qū),邀請大家一同入駐:https://cloud.tencent.com/developer/support-plan?invite_code=3cl4ow56whog8

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請結(jié)合常識(shí)與多方信息審慎甄別。
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

  • 〇、前言 本文共108張圖,流量黨請慎重! 歷時(shí)1個(gè)半月,我把自己學(xué)習(xí)Python基礎(chǔ)知識(shí)的框架詳細(xì)梳理了一遍。 ...
    Raxxie閱讀 19,582評(píng)論 17 410
  • 包(lib)、模塊(module) 在Python中,存在包和模塊兩個(gè)常見概念。 模塊:編寫Python代碼的py...
    清清子衿木子水心閱讀 3,912評(píng)論 0 27
  • 要點(diǎn): 函數(shù)式編程:注意不是“函數(shù)編程”,多了一個(gè)“式” 模塊:如何使用模塊 面向?qū)ο缶幊蹋好嫦驅(qū)ο蟮母拍?、屬性?..
    victorsungo閱讀 1,697評(píng)論 0 6
  • 櫻花盛開的時(shí)節(jié) 我獨(dú)坐岸邊黃昏 風(fēng)從北方吹來 我在風(fēng)里努力找尋櫻花的味道 卻依稀忘記了 那盛開如云的花是怎樣的一種...
    1aec8aca637a閱讀 179評(píng)論 0 0
  • 文/志成哥哥 說句實(shí)話,下了公交車,就看見一片片房子都被拆了,留下一堆廢墟,確實(shí)嚇了一跳。 生活了這么多年的村莊,...
    志成哥哥閱讀 291評(píng)論 4 0

友情鏈接更多精彩內(nèi)容