婷婷综合国产,91蜜桃婷婷狠狠久久综合9色 ,九九九九九精品,国产综合av

主頁 > 知識庫 > Python反射機制實例講解

Python反射機制實例講解

熱門標(biāo)簽:外呼并發(fā)線路 湛江智能外呼系統(tǒng)廠家 百度地圖標(biāo)注沒有了 宿遷星美防封電銷卡 地圖標(biāo)注審核表 西藏房產(chǎn)智能外呼系統(tǒng)要多少錢 ai電銷機器人源碼 長沙高頻外呼系統(tǒng)原理是什么 ai電話機器人哪里好

通常,我們操作對象的屬性或者方法時,是通過點“.”操作符進行的。例如下面的代碼:

class Person:
    type = "mammal"

    def __init__(self, name):
        self.name = name

    def say_hi(self):
        print('Hello, my name is', self.name)

    @staticmethod
    def feed():
        print("Three times per day.")

    @classmethod
    def sleep(cls):
        print("8 hours!")


p = Person('Chunming')
p.say_hi()
print(p.name)

上面代碼的輸出是

Hello, my name is Nikhil
Nikhil

反射是另外一種操作對象屬性和方法的手段,例如:

func = getattr(p, 'say_hi') 
func()
print(getattr(p, "name"))

上面這段代碼的輸出是:

Hello, my name is Nikhil
Nikhil

可見與通過點操作符的結(jié)果一致。

1. 反射的四個函數(shù)

getattr是獲取對象屬性或方法的函數(shù),Python的官方文檔是這樣描述其用法的:

getattr(object, name, value)

返回對象命名屬性的值。name必須是字符串。如果該字符串是對象的屬性之一,則返回該屬性的值。例如, getattr(x, ‘foobar')等同于 x.foobar。如果指定的屬性不存在,且提供了 default值,則返回它,否則觸發(fā) AttributeError。

根據(jù)文檔理解上述代碼,getattr(p, ‘say_hi') 獲取了p對象的say_hi屬性值并賦值給func變量,因為say_hi屬性在Person類中是一個方法,要想調(diào)用這個方法, 需要執(zhí)行func(),getattr(p, “name”) 則是獲取p對象的name屬性。

除了獲取對象屬性和方法的getattr函數(shù),python還內(nèi)置了判斷、設(shè)置、刪除對象屬性和方法的函數(shù),來看看Python官方文檔對這三個函數(shù)的說明:

setattr(object, name, value)

此函數(shù)與 getattr() 兩相對應(yīng)。其參數(shù)為一個對象、一個字符串和一個任意值。字符串指定一個現(xiàn)有屬性或者新增屬性。函數(shù)會將值賦給該屬性,只要對象允許這種操作。例如,setattr(x, ‘foobar', 123) 等價于 x.foobar = 123。

hasattr(object, name)

該實參是一個對象和一個字符串。如果字符串是對象的屬性之一的名稱,則返回 True,否則返回 False。(此功能是通過調(diào)用 getattr(object, name) 看是否有 AttributeError 異常來實現(xiàn)的。)

delattr(object, name)

setattr() 相關(guān)的函數(shù)。實參是一個對象和一個字符串。該字符串必須是對象的某個屬性。如果對象允許,該函數(shù)將刪除指定的屬性。例如 delattr(x, ‘foobar') 等價于 del x.foobar 。

Python中通過getattr、setattr、hasattr和delattr四個函數(shù)操作屬性的機制就是反射。是通過字符串的形式操作對象屬性和方法的機制。下面對p屬性應(yīng)用setattr、hasattr和delattr這三個函數(shù)看看效果:

判斷p對象是否有say_bye屬性和say_hi屬性:

print(hasattr(p, 'say_bye'))  # 輸出False
print(hasattr(p, 'say_hi'))  # 輸出True

給p對象增加say_bye的方法和age屬性:

setattr(p, 'say_bye', say_bye)
setattr(p, 'age', 18)

現(xiàn)在可以訪問這兩個屬性了,通過反射訪問:

bye = getattr(p, 'say_bye')
bye()
print(getattr(p, 'age'))

或者通過點操作符訪問:

p.say_bye()
print(p.age)

刪除age屬性:

delattr(p, 'age')
print(hasattr(p, 'age'))  # 輸出False

2. 類的反射操作

除了對象的反射操作,還有類的反射操作,當(dāng)前模塊的反射操作,還有其他模塊的反射操作,其他包的反射操作。

類的反射操作,指的是對類屬性、類方法或者靜態(tài)方法執(zhí)行反射操作。

獲取類屬性:

t = getattr(Person, 'type')
print(t)  # 輸出mammal
f = getattr(Person, 'feed')
f()  # 輸出Three times per day.
s = getattr(Person, 'sleep')
s() # 8 hours!

判斷類屬性是否存在:

print(hasattr(Person, 'type'))  # 輸出True
print(hasattr(Person, 'name'))  # 輸出False
print(hasattr(Person, 'say_hi')) # 輸出True
print(hasattr(Person, 'sleep'))  # 輸出True
print(hasattr(Person, 'feed'))  # 輸出True

此外,還可以對類添加和刪除屬性和方法。

print(delattr(Person, 'feed'))
print(hasattr(Person, 'feed'))
setattr(Person, 'feed', lambda x: print("Three times per day."))
print(hasattr(Person, 'feed'))

3. 當(dāng)前模塊的反射操作

當(dāng)前模塊也就是當(dāng)前代碼所在的py文件,反射也可以對當(dāng)前模塊中的變量和函數(shù)進行操作。例如某個模塊包含一個al函數(shù),用來判斷迭代對象中每個元素是否都是True,內(nèi)容如下:

import sys

def al(iterable):
    for element in iterable:
        if not element:
            return False
    return True


this_module = sys.modules[__name__]

if hasattr(this_module, 'al'):
    all_true = getattr(this_module, 'al')
    result = all_true([1, 2, 3, 4, True, 0])
    print(result)

通過sys.modules[name]方法獲取當(dāng)前模塊的名稱。getattr第一個參數(shù)是模塊名稱,第二個參數(shù)是想要從模塊中獲取的屬性。

4. 其他模塊反射操作

對import進來的其他模塊中的函數(shù)、屬性、變量進行反射操作。例如,我們導(dǎo)入Python的heapq模塊,這塊模塊提供了堆隊列算法的實現(xiàn),也稱為優(yōu)先隊列算法。下面的代碼是一個實現(xiàn)堆排序的函數(shù)。

import heapq

h = [(5, 'write code'), (7, 'release product'), (1, 'write spec'), (3, 'create tests')]

if hasattr(heapq, 'heapify'):
   heapi = getattr(heapq, 'heapify')  # 獲取heapify屬性
   heapi(h)  # 建堆
   if hasattr(heapq, 'heappop'):
       heapp = getattr(heapq, 'heappop')  # 獲取heappop屬性
       print([heapp(h) for _ in range(len(h))])  # 彈出并從返回堆中最小的項

這里,我們并沒有通過heapq.heapify和heapq.heappop方式調(diào)用heapq模塊中的函數(shù)。而是通過反射達到的同樣的效果。

5. 反射應(yīng)用場景之一

了解了反射中四個函數(shù)的基本用法。那么反射到底有什么用呢?它的應(yīng)用場景是什么呢?答案是,當(dāng)不確定所需要的屬性和函數(shù)是否存在時,可以使用反射。另外一個重要作用是,可以提高代碼的擴展性和可維護性。

假如我們把所有的加密算法都放到一個叫做encryption的模塊中維護 ,并且允許使用這個模塊的用戶添加更多的加密算法到這個模塊中。encryption的模塊內(nèi)容如下:

import hashlib
import os
import sys


def md5(content=None):
    """生成字符串的SHA256值"""
    if content is None:
        return ''
    md5_gen = hashlib.md5()
    md5_gen.update(content.encode('utf-8'))
    md5code = md5_gen.hexdigest()
    return md5code


def sha256(content=None):
    """生成字符串的SHA256值"""
    if content is None:
        return ''
    sha256_gen = hashlib.sha256()
    sha256_gen.update(content.encode('utf-8'))
    sha256code = sha256_gen.hexdigest()
    return sha256code


def sha256_file(filename):
    """生成文件的SHA256值"""
    if not os.path.isfile(filename):
        return ""
    sha256gen = hashlib.sha256()
    size = os.path.getsize(filename)  # 獲取文件大小,單位是Byte
    with open(filename, 'rb') as fd:  # 以二進制方式讀取文件
        while size >= 1024 * 1024:  # 當(dāng)文件大于1MB時分塊讀取文件內(nèi)容
            sha256gen.update(fd.read(1024 * 1024))
            size -= 1024 * 1024
        sha256gen.update(fd.read())
    sha256code = sha256gen.hexdigest()
    return sha256code


def md5_file(filename):
    """生成文件的MD5值"""
    if not os.path.isfile(filename):
        return ""
    md5gen = hashlib.md5()
    size = os.path.getsize(filename)  # 獲取文件大小,單位是Byte
    with open(filename, 'rb') as fd:
        while size >= 1024 * 1024:  # 當(dāng)文件大于1MB時分塊讀取文件內(nèi)容
            md5gen.update(fd.read(1024 * 1024))
            size -= 1024 * 1024
        md5gen.update(fd.read())
    md5code = md5gen.hexdigest()
    return md5code


def encrypt_something(something, algorithm):
    """
    通用加密算法
    :param something: 待加密的內(nèi)容,字符串或者文件
    :param algorithm: 加密算法
    :return:  加密后的內(nèi)容
    """
    result = ""
    if algorithm == "md5":
        result = md5(something)
    elif algorithm == "sh256":
        result = sha256(something)
    elif algorithm == "sh256_file":
        result = sha256_file(something)
    elif algorithm == "md5_file":
        result = md5_file(something)
    return result

其中,encrypt_something函數(shù)提供了通用加密算法,需要調(diào)用者傳入待加密的內(nèi)容和加密算法,這樣當(dāng)調(diào)用者使用encryption.py模塊時,只需導(dǎo)入encrypt_something函數(shù)即可。就像這樣:

import encryption
print(encryption.encrypt_something("learn_python_by_coding", "sh256"))
print(encryption.encrypt_something("learn_python_by_coding", "md5"))

通過分析encrypt_something函數(shù)發(fā)現(xiàn),當(dāng)我們在encryption.py模塊添加更多的加密算法后,就要修改encrypt_something函數(shù),在其中增加新的if分支,隨著加密算法的增加,encrypt_something函數(shù)的分支會越來越多。

學(xué)了反射之后,encrypt_something代碼部分就可以這樣寫:

def encrypt_something(something, algorithm):
    """
    通用加密算法
    :param something: 待加密的內(nèi)容,字符串或者文件
    :param algorithm: 加密算法
    :return:  加密后的內(nèi)容
    """
    this_module = sys.modules[__name__]
    if hasattr(this_module, algorithm):
        algorithm = getattr(this_module, algorithm)
        result = algorithm(something)
    else:
        raise ValueError("Not support {} algorithm".format(algorithm))
    return result

相比前面的采用if分支語句方式,反射更加簡潔明了,可維護性更強,要想增加新的加密方法,只需要在encryption.py模塊添加更多的加密算法即可,encrypt_something代碼不需要任何變更。

6. 反射應(yīng)用場景之二

再看一個基于Pytest測試框架的測試腳本中應(yīng)用反射的例子,比如conftest文件內(nèi)容如下:

# content of conftest.py
import pytest
import smtplib


@pytest.fixture(scope="module")
def smtp_connection(request):
    server = getattr(request.module, "smtpserver", "smtp.gmail.com")
    smtp_connection = smtplib.SMTP(server, 587, timeout=5)
    yield smtp_connection
    print("finalizing {} ({})".format(smtp_connection, server))
    smtp_connection.close()

request.module 是所有測試腳本,就是那些以_test結(jié)尾或者test_開頭的py文件。

server = getattr(request.module, "smtpserver", "smtp.gmail.com") 

含義就是從測試腳本文件中找smtpserver屬性,如果找不到,默認使用smtp.gmail.com作為smtpserver的值。如果測試腳本文件有這個屬性,則使用測試腳本中的值,例如下面這個測試腳本,smtpserver則會使用mail.python.org這個值:

# content of test_anothersmtp.py

smtpserver = "mail.python.org"  # will be read by smtp fixture


def test_showhelo(smtp_connection):
    assert 0, smtp_connection.helo()

7. 總結(jié)

在很多開源框架中普遍采用,是提高可維護性和擴展性的利器。如果工作中也涉及到框架開發(fā),反射一定會助一臂之力,,希望大家以后多多支持腳本之家!

您可能感興趣的文章:
  • python對驗證碼降噪的實現(xiàn)示例代碼
  • 爬蟲Python驗證碼識別入門
  • Python機器學(xué)習(xí)入門(一)序章
  • 6個Python辦公黑科技,助你提升工作效率
  • Python機器學(xué)習(xí)入門(三)之Python數(shù)據(jù)準(zhǔn)備
  • 用python寫個顏值評分器篩選最美主播
  • Python代碼實現(xiàn)粒子群算法圖文詳解
  • 我用Python做個AI出牌器斗地主把把贏
  • python通過PyQt5實現(xiàn)登錄界面的示例代碼
  • Python圖片驗證碼降噪和8鄰域降噪

標(biāo)簽:林芝 海南 盤錦 漯河 寧夏 南平 大同 普洱

巨人網(wǎng)絡(luò)通訊聲明:本文標(biāo)題《Python反射機制實例講解》,本文關(guān)鍵詞  Python,反射,機制,實例,講解,;如發(fā)現(xiàn)本文內(nèi)容存在版權(quán)問題,煩請?zhí)峁┫嚓P(guān)信息告之我們,我們將及時溝通與處理。本站內(nèi)容系統(tǒng)采集于網(wǎng)絡(luò),涉及言論、版權(quán)與本站無關(guān)。
  • 相關(guān)文章
  • 下面列出與本文章《Python反射機制實例講解》相關(guān)的同類信息!
  • 本頁收集關(guān)于Python反射機制實例講解的相關(guān)信息資訊供網(wǎng)民參考!
  • 推薦文章
    婷婷综合国产,91蜜桃婷婷狠狠久久综合9色 ,九九九九九精品,国产综合av
    色国产综合视频| 3d动漫精品啪啪一区二区竹菊| 一区二区三区加勒比av| 久久精品欧美一区二区三区麻豆| 欧美精选午夜久久久乱码6080| 欧美日韩成人在线| 欧美三级电影精品| 日韩欧美国产午夜精品| 久久色中文字幕| 亚洲视频一区在线| 亚洲一区在线观看免费观看电影高清| 国产女同互慰高潮91漫画| 高清久久久久久| 五月婷婷综合网| av激情亚洲男人天堂| 91视频在线观看| 久久精品国产精品青草| 免费看欧美女人艹b| 久久精品久久精品| 久久精品欧美一区二区三区麻豆| 国产精品夜夜爽| 精品一区二区三区日韩| www.色精品| 欧美男男青年gay1069videost| 欧美日韩综合不卡| 久久久亚洲高清| 成av人片一区二区| 色婷婷综合久久久中文一区二区 | 另类小说图片综合网| 激情深爱一区二区| 色综合中文字幕| 国产欧美视频在线观看| 欧美一区二区成人| 亚洲国产精品成人久久综合一区| 一区二区三区四区乱视频| 欧美国产一区在线| 26uuu国产一区二区三区| 亚洲一区二区三区视频在线| 国产成人日日夜夜| 久久久久久亚洲综合| 免费观看在线综合色| 日韩一级成人av| 麻豆精品在线播放| 欧美精品一区二区三区在线播放 | 波多野结衣中文一区| 欧美亚洲自拍偷拍| 国内外精品视频| ...中文天堂在线一区| 欧美色手机在线观看| 青青青伊人色综合久久| 欧美va日韩va| 一区在线播放视频| 粉嫩av一区二区三区在线播放| 日韩午夜激情免费电影| 黄一区二区三区| 亚洲精品国产无天堂网2021| 欧美性videosxxxxx| 国产麻豆视频一区二区| 欧美成人video| 色综合色狠狠天天综合色| 日韩一区欧美一区| 欧美中文一区二区三区| 久久电影网电视剧免费观看| 日韩欧美一二区| 99国产精品视频免费观看| 亚洲成av人影院| 亚洲欧美怡红院| 久久久美女艺术照精彩视频福利播放| 色婷婷久久99综合精品jk白丝| 色噜噜狠狠色综合欧洲selulu| 丝袜美腿亚洲色图| 亚洲日穴在线视频| 99久免费精品视频在线观看| 99精品在线观看视频| 日本欧美一区二区在线观看| 一级精品视频在线观看宜春院| 久久精品这里都是精品| 91精品国产综合久久小美女| 欧美揉bbbbb揉bbbbb| 色一区在线观看| 色欧美日韩亚洲| 欧美综合在线视频| 成人免费看黄yyy456| 九九视频精品免费| 黑人巨大精品欧美黑白配亚洲| 亚洲一区在线观看免费观看电影高清| 久久精品日产第一区二区三区高清版| 日韩一区二区精品葵司在线| 99精品欧美一区二区蜜桃免费| 国产精品亚洲а∨天堂免在线| 成人一区二区三区视频在线观看| 亚洲人妖av一区二区| 亚洲精品欧美激情| 91蜜桃免费观看视频| 成年人国产精品| 欧美日韩黄色影视| 97久久久精品综合88久久| 国产成人精品免费网站| 欧美午夜在线一二页| 久久综合色婷婷| 婷婷丁香激情综合| hitomi一区二区三区精品| 欧美性一级生活| 亚洲欧美影音先锋| 欧美电视剧免费观看| 国产精品免费视频一区| 亚洲影院久久精品| 国产91丝袜在线18| 欧美日韩一区 二区 三区 久久精品| 亚洲美女屁股眼交3| 欧美日韩成人一区二区| 国产精品视频免费看| 美日韩一区二区| 欧美综合一区二区| 国产欧美视频一区二区三区| 国产精品久久久久三级| 婷婷中文字幕综合| 国产在线精品一区二区夜色 | 日本二三区不卡| 欧美丝袜丝交足nylons图片| 国产精品久久久久久久久快鸭| 一区二区成人在线| 精品无人码麻豆乱码1区2区| 色偷偷久久一区二区三区| 2021中文字幕一区亚洲| 亚洲自拍另类综合| 91麻豆国产福利精品| 亚洲欧美偷拍另类a∨色屁股| 成人性色生活片免费看爆迷你毛片| 久久日韩粉嫩一区二区三区| 久久 天天综合| 久久网站最新地址| 韩国精品在线观看| 日本一区二区三级电影在线观看| 岛国精品在线观看| 亚洲一区中文在线| 久久久综合激的五月天| 91欧美一区二区| 中文字幕在线观看不卡| 国产成人精品aa毛片| 欧美国产精品v| 欧美精品在欧美一区二区少妇| 日韩和欧美一区二区三区| 9191成人精品久久| 972aa.com艺术欧美| 亚洲国产aⅴ成人精品无吗| 337p粉嫩大胆色噜噜噜噜亚洲| 国产精品99久久久| 一区二区三区小说| 亚洲国产精品成人综合 | 3d动漫精品啪啪一区二区竹菊| 久久成人免费电影| 亚洲一区在线观看免费| 2023国产精品自拍| 欧美老年两性高潮| 免费的成人av| 视频一区视频二区中文| 日本一区二区高清| 精品国精品国产尤物美女| 精品视频在线免费观看| 暴力调教一区二区三区| 风间由美一区二区三区在线观看 | 国产精品国产三级国产a| 精品久久久久久久久久久久久久久久久| 亚洲欧洲日本在线| 久久麻豆一区二区| 欧美性欧美巨大黑白大战| 亚洲国产美女搞黄色| 国产精品的网站| 亚洲男人的天堂一区二区| 亚洲国产综合在线| 亚洲电影一区二区| 老司机精品视频在线| 久久成人18免费观看| 国产精品美女久久福利网站| 国产精品每日更新在线播放网址| 中文字幕免费在线观看视频一区| 在线观看视频一区二区欧美日韩| av在线免费不卡| 91亚洲国产成人精品一区二区三| 波多野结衣在线一区| 在线一区二区三区四区五区| 91麻豆精品秘密| 欧美图区在线视频| 欧美一区欧美二区| 日韩一二在线观看| 国产精品不卡一区二区三区| 天堂av在线一区| 亚洲久草在线视频| 日本亚洲视频在线| 成人小视频免费在线观看| 成人av电影观看| 在线视频观看一区| 91麻豆精品国产| 精品少妇一区二区三区在线播放 | 97se亚洲国产综合自在线| 欧美成人综合网站| 日本午夜一区二区| 一本色道久久综合亚洲aⅴ蜜桃| 久久亚洲私人国产精品va媚药|