反射 反射機制就是在運行時,,動態(tài)的確定對象的類型,,并可以通過字符串調用對象屬性、方法,、導入模塊,,是一種基于字符串的事件驅動。
python是一門解釋型語言,,因此對于反射機制支持很好,。在python中支持反射機制的函數(shù)有getattr()、setattr(),、delattr(),、exec()、eval(),、__import__,,這些函數(shù)都可以執(zhí)行字符串。 eval 計算指定表達式的值,。它只能執(zhí)行單個表達式,,而不能是復雜的代碼邏輯。而且不能是賦值表達式,。 單個表達式: a = "12 + 43" 復雜表達式: a = "print(12 + 43); print(1111)" 賦值: a = 1 通常我們使用eval的時候,,主要是使用它的返回值,獲取表達式計算出的值 exec 執(zhí)行復雜表達式,,返回值永遠都是None b = exec("aa = 21") 執(zhí)行復雜語句: a = '''ret = [] 導入模塊: # 導入模塊 導入模塊這個功能就非常厲害了,,這樣我們就可以動態(tài)的創(chuàng)建各種模塊類。
再看一下下面的例子: class Base: 如果我們想通過字符串來調用a對象的test方法,,應該怎么做呢,如果要獲取返回值,,那么可以使用 b = eval("a.test()") 輸出: test Base::test 如果不需要獲取返回值,,那么可以使用exec,,exec("a.test()"),輸出:test 雖然我們可以使用eval和exec來執(zhí)行以上代碼,,但是這種方式有一個缺陷,,假如這個屬性是不存在的,那么這種調用就會報錯,。那么做好的方式是什么呢,?先判斷屬性是否存在,如果存在就調用,,不存在就不調用,,python為我們提供了一套方法:hasattr、getattr,、setattr,、delattr hasattr def hasattr(*args, **kwargs): # real signature unknown 通過源碼注釋我們知道,它返回對象是否具有指定名稱的屬性,。而且它是通過調用getattr并捕獲AttributeError異常來判斷的,。就像上面的屬性調用,我們就可以使用hasattr(a, "test")來判斷,,通過源碼注釋我們也可以思考一下,,eval這種是不是也可以實現(xiàn)這種方法呢? def has_attr(obj, name): 但是這種方式是有缺陷的,,因為test輸出了兩次,,因為我們調用了兩次test(),這跟我們想要的效果不一樣,。如果用hasattr呢,,這個函數(shù)就不會在判斷的時候調用一次了。 getattr() 有了判斷屬性是否存在的函數(shù),,那么就得有獲取屬性的函數(shù)了 def getattr(object, name, default=None): # known special case of getattr 從源碼注釋我們就能知道獲取object對象的名為name的屬性,,想到與object.name,如果提供了default參數(shù),,那么當屬性不存在的時候,,就會返回默認值。同樣是上面的例子: a = Base() 從例子中我們可以看出,,hasattr并沒有調用test函數(shù),,而且getattr獲取到的是函數(shù)對象,也沒有調用它,,通過我們主動執(zhí)行func()才執(zhí)行了a.test()函數(shù),,這樣相比于exec和eval就靈活了許多。 setattr 判斷和獲取屬性有了,,那么設置屬性也是需要的 def setattr(x, y, v): # real signature unknown; restored from __doc__ 將一個特殊值設置給object對象的name屬性,,相當于x.y = v class Base: 雖然setattr(a, "age", 32)等于a.age=32,但是我們不要忘了,,這是通過一個字符串來增加的屬性,。 判斷、獲取,、增加都有了,當然還有刪除delattr,,這個我們就不詳述了,,接下來我們要看一個比較重要的方法。 import 在學習exec的時候,,我們有一個例子,,導入配置文件exec("import config"),針對這種方式python也為我們提供了更好的方法,。 def __import__(name, globals=None, locals=None, fromlist=(), level=0): # real signature unknown; restored from __doc__ 在這里我們最需要關注的是formlist參數(shù),,先看一個簡單的例子: a = __import__("config") config是一個py腳本-config.py,內部有一個變量KEYWORD,,我們要通過其他py模塊來導入這個文件,,使用__import__我們就可以把它導入為一個對象,然后使用對象的方式去調用,,而不是一直用exec字符串的形式去調用,。上面我們說了formlist這個參數(shù)需要關注,為什么呢,?我們新增了一個模塊:comm,。模塊內有一個腳本function.py # function.py 我們現(xiàn)在想通過動態(tài)引入的方式調用comm_function函數(shù),那么按照上面的方式來 a = __import__("comm.function") 結果輸出: Traceback (most recent call last): 意思是comm模塊沒有comm_function這個屬性,,為什么是comm模塊而不是function呢,?我們可以打印一下模塊的引入名稱print(a.__name__),打印的結果是comm,,就是說我們通過上面的方式只是引入comm,,而不是function。其實通過源碼注釋我們就知道了,,__import__(A.B),,如果fromlist為空,返回的是A包,,如果不為空,,則返回其子包B。修改一下我們的代碼: a = __import__("comm.function", fromlist=True) 引入的模塊和執(zhí)行函數(shù)都正確了,,符合了我們的預期要求,。 總結 通過以上的函數(shù)學習,,其中有常用的,也有不常用的,,但是這些函數(shù)在我們進行框架設計時是必不可少的,,尤其是__import__,接下來我們還會繼續(xù)看框架設計中最重要的一個概念--元編程,。學完了這些概念就可以設計框架了,。開玩笑的,哪有那么簡單,。閱讀源碼是一種增長知識的最快捷方式,,但是前提是基礎一定要打好。否則看源碼是一頭霧水,。我們整理完這些概念后,,在找?guī)讉€源碼庫看看,學習一下里面的設計理念,。 |
|
來自: 千鋒Python學堂 > 《Python基礎教程分享》