inspect
--- 檢查對象?
源代碼: Lib/inspect.py
inspect
模塊提供了一些有用的函數幫助獲取對象的信息,例如模塊、類(lèi)、方法、函數、回溯、幀對象以及代碼對象。例如它可以幫助你檢查類(lèi)的內容,獲取某個(gè)方法的源代碼,取得并格式化某個(gè)函數的參數列表,或者獲取你需要顯示的回溯的詳細信息。
該模塊提供了4種主要的功能:類(lèi)型檢查、獲取源代碼、檢查類(lèi)與函數、檢查解釋器的調用堆棧。
類(lèi)型和成員?
getmembers()
函數獲取對象的成員,例如類(lèi)或模塊。函數名以"is"開(kāi)始的函數主要作為 getmembers()
的第2個(gè)參數使用。它們也可用于判定某對象是否有如下的特殊屬性:
類(lèi)型 |
屬性 |
描述 |
---|---|---|
module -- 模塊 |
__doc__ |
文檔字符串 |
__file__ |
文件名(內置模塊沒(méi)有文件名) |
|
class -- 類(lèi) |
__doc__ |
文檔字符串 |
__name__ |
類(lèi)定義時(shí)所使用的名稱(chēng) |
|
__qualname__ |
qualified name -- 限定名稱(chēng) |
|
__module__ |
該類(lèi)型被定義時(shí)所在的模塊的名稱(chēng) |
|
method -- 方法 |
__doc__ |
文檔字符串 |
__name__ |
該方法定義時(shí)所使用的名稱(chēng) |
|
__qualname__ |
qualified name -- 限定名稱(chēng) |
|
__func__ |
實(shí)現該方法的函數對象 |
|
__self__ |
該方法被綁定的實(shí)例,若沒(méi)有綁定則為 |
|
__module__ |
定義此方法的模塊的名稱(chēng) |
|
function -- 函數 |
__doc__ |
文檔字符串 |
__name__ |
用于定義此函數的名稱(chēng) |
|
__qualname__ |
qualified name -- 限定名稱(chēng) |
|
__code__ |
包含已編譯函數的代碼對象 bytecode |
|
__defaults__ |
所有位置或關(guān)鍵字參數的默認值的元組 |
|
__kwdefaults__ |
保存僅關(guān)鍵字參數的所有默認值的映射 |
|
__globals__ |
此函數定義所在的全局命名空間 |
|
__builtins__ |
builtins 命名空間 |
|
__annotations__ |
參數名稱(chēng)到注解的映射;保留鍵 |
|
__module__ |
此函數定義所在的模塊名稱(chēng) |
|
traceback -- 回溯 |
tb_frame |
此層的幀對象 |
tb_lasti |
在字節碼中最后嘗試的指令的索引 |
|
tb_lineno |
當前行在 Python 源代碼中的行號 |
|
tb_next |
下一個(gè)內部回溯對象(由本層調用) |
|
frame -- 幀 |
f_back |
下一個(gè)外部幀對象(此幀的調用者) |
f_builtins |
此幀執行時(shí)所在的 builtins 命名空間 |
|
f_code |
在此幀中執行的代碼對象 |
|
f_globals |
此幀執行時(shí)所在的全局命名空間 |
|
f_lasti |
在字節碼中最后嘗試的指令的索引 |
|
f_lineno |
當前行在 Python 源代碼中的行號 |
|
f_locals |
此幀所看到的局部命名空間 |
|
f_trace |
此幀的追蹤函數,或``None`` |
|
code -- 代碼 |
co_argcount |
參數數量(不包括僅關(guān)鍵字參數、* 或 ** 參數) |
co_code |
字符串形式的原始字節碼 |
|
co_cellvars |
單元變量名稱(chēng)的元組(通過(guò)包含作用域引用) |
|
co_consts |
字節碼中使用的常量元組 |
|
co_filename |
創(chuàng )建此代碼對象的文件的名稱(chēng) |
|
co_firstlineno |
第一行在Python源代碼中的行號 |
|
co_flags |
|
|
co_lnotab |
編碼的行號到字節碼索引的映射 |
|
co_freevars |
自由變量的名字組成的元組(通過(guò)函數閉包引用) |
|
co_posonlyargcount |
僅限位置參數的數量 |
|
co_kwonlyargcount |
僅限關(guān)鍵字參數的數量(不包括 ** 參數) |
|
co_name |
定義此代碼對象的名稱(chēng) |
|
co_qualname |
fully-qualified name with which this code object was defined |
|
co_names |
tuple of names other than arguments and function locals |
|
co_nlocals |
局部變量的數量 |
|
co_stacksize |
需要虛擬機堆??臻g |
|
co_varnames |
參數名和局部變量的元組 |
|
generator -- 生成器 |
__name__ |
名稱(chēng) |
__qualname__ |
qualified name -- 限定名稱(chēng) |
|
gi_frame |
frame -- 幀 |
|
gi_running |
生成器在運行嗎? |
|
gi_code |
code -- 代碼 |
|
gi_yieldfrom |
通過(guò)``yield from``迭代的對象,或``None`` |
|
coroutine -- 協(xié)程 |
__name__ |
名稱(chēng) |
__qualname__ |
qualified name -- 限定名稱(chēng) |
|
cr_await |
正在等待的對象,或``None`` |
|
cr_frame |
frame -- 幀 |
|
cr_running |
這個(gè)協(xié)程正在運行嗎? |
|
cr_code |
code -- 代碼 |
|
cr_origin |
協(xié)程被創(chuàng )建的位置,或``None``。參見(jiàn) |
|
builtin |
__doc__ |
文檔字符串 |
__name__ |
此函數或方法的原始名稱(chēng) |
|
__qualname__ |
qualified name -- 限定名稱(chēng) |
|
__self__ |
方法綁定到的實(shí)例,或``None`` |
在 3.5 版更改: 為生成器添加``__qualname__``和``gi_yieldfrom``屬性。
生成器的``__name__``屬性現在由函數名稱(chēng)設置,而不是代碼對象名稱(chēng),并且現在可以被修改。
在 3.7 版更改: 為協(xié)程添加``cr_origin``屬性。
在 3.10 版更改: 為函數添加``__builtins__``屬性。
- inspect.getmembers(object[, predicate])?
返回一個(gè)對象上的所有成員,組成以
(名稱(chēng), 值)
對為元素的列表,按名稱(chēng)排序。如果提供了可選的 predicate 參數(會(huì )對每個(gè)成員的值
對象進(jìn)行一次調用),則僅包含該斷言為真的成員。備注
如果要讓
getmembers()
返回元類(lèi)中定義的類(lèi)屬性,那么實(shí)參須為一個(gè)類(lèi)且這些屬性在元類(lèi)的自定義方法__dir__()
中列出。
- inspect.getmembers_static(object[, predicate])?
Return all the members of an object in a list of
(name, value)
pairs sorted by name without triggering dynamic lookup via the descriptor protocol, __getattr__ or __getattribute__. Optionally, only return members that satisfy a given predicate.備注
getmembers_static()
may not be able to retrieve all members that getmembers can fetch (like dynamically created attributes) and may find members that getmembers can't (like descriptors that raise AttributeError). It can also return descriptor objects instead of instance members in some cases.3.11 新版功能.
- inspect.getmodulename(path)?
返回由文件名 path 表示的模塊名字,但不包括外層的包名。文件擴展名會(huì )檢查是否在
importlib.machinery.all_suffixes()
列出的條目中。如果符合,則文件路徑的最后一個(gè)組成部分會(huì )去掉后綴名后被返回;否則返回``None``。值得注意的是,這個(gè)函數 僅 返回可以作為 Python 模塊的名字,而有可能指向一個(gè) Python 包的路徑仍然會(huì )返回``None``。
在 3.3 版更改: 該函數直接基于
importlib
。
- inspect.ismodule(object)?
當該對象是一個(gè)模塊時(shí)返回``True``。
- inspect.isclass(object)?
當該對象是一個(gè)類(lèi)時(shí)返回``True``,無(wú)論是內置類(lèi)或者 Python 代碼中定義的類(lèi)。
- inspect.ismethod(object)?
當該對象是一個(gè) Python 寫(xiě)成的綁定方法時(shí)返回``True``。
- inspect.isgeneratorfunction(object)?
當該對象是一個(gè) Python 生成器函數時(shí)返回``True``。
在 3.8 版更改: 對于使用
functools.partial()
封裝的函數,如果被封裝的函數是一個(gè) Python 生成器函數,現在也會(huì )返回``True``。
- inspect.isgenerator(object)?
當該對象是一個(gè)生成器時(shí)返回``True``。
- inspect.iscoroutinefunction(object)?
當該對象是一個(gè) 協(xié)程函數 (通過(guò)
async def
語(yǔ)法定義的函數)時(shí)返回``True``。3.5 新版功能.
在 3.8 版更改: 對于使用
functools.partial()
封裝的函數,如果被封裝的函數是一個(gè) 協(xié)程函數 ,現在也會(huì )返回``True``。
- inspect.iscoroutine(object)?
當該對象是一個(gè)由
async def
函數創(chuàng )建的 協(xié)程 時(shí)返回``True``。3.5 新版功能.
- inspect.isawaitable(object)?
如果該對象可以在
await
表達式中使用時(shí)返回``True``。也可以用于區分基于生成器的協(xié)程和常規的生成器:
def gen(): yield @types.coroutine def gen_coro(): yield assert not isawaitable(gen()) assert isawaitable(gen_coro())
3.5 新版功能.
- inspect.isasyncgenfunction(object)?
如果該對象是一個(gè) 異步生成器 函數則返回``True``,例如:
>>> async def agen(): ... yield 1 ... >>> inspect.isasyncgenfunction(agen) True
3.6 新版功能.
在 3.8 版更改: 對于使用
functools.partial()
封裝的函數,如果被封裝的函數是一個(gè) 異步生成器 ,現在也會(huì )返回``True``。
- inspect.istraceback(object)?
如果該對象是一個(gè)回溯則返回
True
。
- inspect.isframe(object)?
如果該對象是一個(gè)幀對象則返回
True
。
- inspect.iscode(object)?
如果該對象是一個(gè)代碼對象則返回
True
。
- inspect.isbuiltin(object)?
如果該對象是一個(gè)內置函數或一個(gè)綁定的內置方法,則返回
True
。
- inspect.ismethodwrapper(object)?
Return
True
if the type of object is aMethodWrapperType
.These are instances of
MethodWrapperType
, such as__str__()
,__eq__()
and__repr__()
- inspect.isroutine(object)?
如果該對象是一個(gè)用戶(hù)定義的或內置的函數或者方法,則返回
True
。
- inspect.isabstract(object)?
如果該對象是一個(gè)抽象基類(lèi)則返回
True
。
- inspect.ismethoddescriptor(object)?
如果該對象是一個(gè)方法描述器,但
ismethod()
、isclass()
、isfunction()
及isbuiltin()
均不為真,則返回True
。例如,該函數對于
int.__add__
為真。一個(gè)通過(guò)此測試的對象可以有__get__()
方法,但不能有__set__()
方法,除此以外的屬性集合是可變的。一個(gè)__name__
屬性通常是合理的,而__doc__
屬性常常也是。即使是通過(guò)描述器實(shí)現的函數,如果通過(guò)其他某一個(gè)測試,則
ismethoddescriptor()
測試則會(huì )返回False
。這單純是因為其他測試提供了更多保證。比如,當一個(gè)對象通過(guò)ismethod()
測試時(shí),你可以使用__func__
等屬性。
- inspect.isdatadescriptor(object)?
如果該對象是一個(gè)數據描述器則返回
True
。數據描述器擁有
__set__
或__delete__
方法。比如屬性(通過(guò) Python 定義)、getset 和成員。后二者是通過(guò) C 定義的,并且也有對應的更具體的測試,并且在不同的 Python 實(shí)現中也是健壯的。典型地,數據描述器會(huì )擁有__name__
和__doc__
屬性(屬性、getset 和成員都包含這兩個(gè)屬性),但這并無(wú)保證。
- inspect.isgetsetdescriptor(object)?
如果該對象是一個(gè) getset 描述器則返回
True
。CPython implementation detail: getset 是在擴展模塊中通過(guò)
PyGetSetDef
結構體定義的屬性。對于不包含這種類(lèi)型的 Python 實(shí)現,這個(gè)方法將永遠返回False
。
- inspect.ismemberdescriptor(object)?
如果該對象是一個(gè)成員描述器則返回
True
。CPython implementation detail: 成員描述器是在擴展模塊中通過(guò)
PyMemberDef
結構體定義的屬性。對于不包含這種類(lèi)型的 Python 實(shí)現,這個(gè)方法將永遠返回False
。
獲取源代碼?
- inspect.getdoc(object)?
Get the documentation string for an object, cleaned up with
cleandoc()
. If the documentation string for an object is not provided and the object is a class, a method, a property or a descriptor, retrieve the documentation string from the inheritance hierarchy. ReturnNone
if the documentation string is invalid or missing.在 3.5 版更改: 文檔字符串沒(méi)有被重寫(xiě)的話(huà)現在會(huì )被繼承。
- inspect.getcomments(object)?
任意多行注釋作為單一一個(gè)字符串返回。對于類(lèi)、函數和方法,選取緊貼在該對象的源代碼之前的注釋?zhuān)粚τ谀K,選取 Python 源文件頂部的注釋。如果對象的源代碼不可獲得,返回
None
。這可能是因為對象是 C 語(yǔ)言中或者是在交互式命令行中定義的。
- inspect.getfile(object)?
返回定義了這個(gè)對象的文件名(包括文本文件或二進(jìn)制文件)。如果該對象是一個(gè)內置模塊、類(lèi)或函數則會(huì )失敗并引發(fā)一個(gè)
TypeError
。
- inspect.getmodule(object)?
Try to guess which module an object was defined in. Return
None
if the module cannot be determined.
- inspect.getsourcefile(object)?
Return the name of the Python source file in which an object was defined or
None
if no way can be identified to get the source. This will fail with aTypeError
if the object is a built-in module, class, or function.
- inspect.getsourcelines(object)?
返回一個(gè)源代碼行的列表和對象的起始行。實(shí)參可以是一個(gè)模塊、類(lèi)、方法、函數、回溯、幀或者代碼對象。與該對象相關(guān)的源代碼會(huì )按行構成一個(gè)列表,并且會(huì )有一個(gè)行號表示其中第一行代碼出現在源文件的第幾行。如果源代碼不能被獲取,則會(huì )引發(fā)一個(gè)
OSError
。
- inspect.getsource(object)?
返回對象的源代碼文本。實(shí)參可以是一個(gè)模塊、類(lèi)、方法、函數、回溯、幀或者代碼對象。源代碼會(huì )作為單一一個(gè)字符串被返回。如果源代碼不能被獲取,則會(huì )引發(fā)一個(gè)
OSError
。
- inspect.cleandoc(doc)?
清理文檔字符串中為對齊當前代碼塊進(jìn)行的縮進(jìn)
第一行的所有前綴空白符會(huì )被移除。從第二行開(kāi)始所有可以被統一去除的空白符也會(huì )被去除。之后,首尾的空白行也會(huì )被移除。同時(shí),所有制表符會(huì )被展開(kāi)到空格。
使用 Signature 對象對可調用對象進(jìn)行內省?
3.3 新版功能.
Signature 對象代表了一個(gè)可調用對象的調用簽名和返回值標注。要獲取一個(gè) Signature 對象,使用 signature()
函數。
- inspect.signature(callable, *, follow_wrapped=True, globals=None, locals=None, eval_str=False)?
返回給定的
callable
的一個(gè)Signature
對象:>>> from inspect import signature >>> def foo(a, *, b:int, **kwargs): ... pass >>> sig = signature(foo) >>> str(sig) '(a, *, b:int, **kwargs)' >>> str(sig.parameters['b']) 'b:int' >>> sig.parameters['b'].annotation <class 'int'>
接受各類(lèi)的 Python 可調用對象,包括單純的函數、類(lèi),到
functools.partial()
對象。對于在字符串化標注模塊(
from __future__ import annotations
)中定義的對象,signature()
會(huì )嘗試使用inspect.get_annotations()
自動(dòng)地反字符串化這些標注。參數global
locals
和eval_str
會(huì )在分析標注時(shí)被傳入inspect.get_annotations()
。如何運用這些參數請參見(jiàn)inspect.get_annotations()
的文檔。如果沒(méi)有簽名可以提供,則引發(fā)
ValueError
;如果不支持該對象類(lèi)型,則引發(fā)TypeError
。同時(shí),如果標注被反字符串化,并且eval_str
不是假值,則eval()
會(huì )被調用以反字符串化,此時(shí)有可能引發(fā)任何種類(lèi)的異常。函數簽名中的斜杠(/)表示在它之前的參數是僅限位置的。詳見(jiàn) 編程常見(jiàn)問(wèn)題中關(guān)于僅限位置參數的條目
3.5 新版功能:
follow_wrapped
形參。傳遞False
來(lái)獲得特定關(guān)于callable
的簽名(callable.__wrapped__
將不會(huì )用來(lái)解包被修飾的可調用對象)。3.10 新版功能: 添加參數
globals
、locals
和eval_str
。備注
一些可調用對象可能在特定 Python 實(shí)現中無(wú)法被內省。例如,在 CPython 中,部分通過(guò) C 語(yǔ)言定義的內置函數不提供關(guān)于其參數的元數據。
- class inspect.Signature(parameters=None, *, return_annotation=Signature.empty)?
一個(gè) Signature 對象代表了一個(gè)函數的調用簽名和返回值標注。對于函數接受的每個(gè)參數,它對應地在自身的
parameters
容器中存儲一個(gè)Parameter
對象。可選參數 parameters 是一個(gè)
Parameter
對象組成的序列,它會(huì )在之后被驗證不存在名字重復的參數,并且參數處于正確的順序,即僅限位置參數最前,之后緊接著(zhù)可位置可關(guān)鍵字參數,并且有默認值參數在無(wú)默認值參數之前。可選參數 return_annotation 可以是任意 Python 對象,是該可調用對象中“返回值”的標注。
Signature 對象是 不可變 的。使用
Signature.replace()
來(lái)構造一個(gè)修改后的副本。在 3.5 版更改: Signature 對象既可封存(pickle)又可哈希。
- empty?
該類(lèi)的一個(gè)特殊標記來(lái)明確指出返回值標注缺失。
- parameters?
一個(gè)參數名字到對應
Parameter
對象的有序映射。參數以嚴格的定義順序出現,包括僅關(guān)鍵字參數。在 3.7 版更改: Python 從 3.7 版起才顯式地保證了它保持僅關(guān)鍵字參數的定義順序,盡管實(shí)踐上在 Python 3 中一直保持了這個(gè)順序。
- return_annotation?
可調用對象的“返回值”標注。如果可調用對象沒(méi)有“返回值”標注,這個(gè)屬性會(huì )被設置為
Signature.empty
。
- bind(*args, **kwargs)?
構造一個(gè)位置和關(guān)鍵字實(shí)參到形參的映射。如果
*args
和**kwargs
符合簽名,則返回一個(gè)BoundArguments
;否則引發(fā)一個(gè)TypeError
。
- bind_partial(*args, **kwargs)?
與
Signature.bind()
作用方式相同,但允許省略部分必要的參數(模仿functools.partial()
的行為)。返回BoundArguments
,或者在傳入參數不符合簽名的情況下,引發(fā)一個(gè)TypeError
。
- replace(*[, parameters][, return_annotation])?
基于 replace 函數被調用的目標,創(chuàng )建一個(gè)新的 Signature 實(shí)例??梢詡鬟f不同的
parameters
和/或return_annotation
來(lái)覆蓋原本簽名的對應的屬性。要從復制的 Signature 中移除 return_annotation,可以傳入Signature.empty
。>>> def test(a, b): ... pass >>> sig = signature(test) >>> new_sig = sig.replace(return_annotation="new return anno") >>> str(new_sig) "(a, b) -> 'new return anno'"
- classmethod from_callable(obj, *, follow_wrapped=True, globalns=None, localns=None)?
返回給定可調用對象
obj
的Signature
對象(或其子類(lèi))。傳入follow_wrapped=False
來(lái)直接得到obj
的簽名,而不通過(guò)__wrapped__
鏈來(lái)解析包裝。在解析標注時(shí)會(huì )將globalns
和localns
用作命名空間。該函數簡(jiǎn)化了創(chuàng )建
Signature
的子類(lèi)的過(guò)程:class MySignature(Signature): pass sig = MySignature.from_callable(min) assert isinstance(sig, MySignature)
3.5 新版功能.
3.10 新版功能: 添加了參數
globalns
和localns
。
- class inspect.Parameter(name, kind, *, default=Parameter.empty, annotation=Parameter.empty)?
Parameter 對象是 不可變 的。不要直接修改 Parameter 對象,而是通過(guò)
Parameter.replace()
來(lái)創(chuàng )建一個(gè)修改后的副本。在 3.5 版更改: Parameter 對象既可封存(pickle)又可哈希。
- empty?
該類(lèi)的一個(gè)特殊標記來(lái)明確指出默認值和標注的缺失。
- name?
參數的名字字符串。這個(gè)名字必須是一個(gè)合法的 Python 標識符。
CPython implementation detail: CPython 會(huì )為用于實(shí)現推導式和生成器表達式的代碼對象構造形如
.0
的隱式形參名。在 3.6 版更改: 這些形參名會(huì )被此模塊暴露為形如
implicit0
一樣的名字。
- default?
該參數的默認值。如果該參數沒(méi)有默認值,這個(gè)屬性會(huì )被設置為
Parameter.empty
。
- annotation?
該參數的標注。如果該參數沒(méi)有標注,這個(gè)屬性會(huì )被設置為
Parameter.empty
。
- kind?
描述實(shí)參值會(huì )如何綁定到形參??赡艿娜≈担梢酝ㄟ^(guò)
Parameter
獲得,比如Parameter.KEYWORD_ONLY
):名稱(chēng)
含意
POSITIONAL_ONLY
值必須以位置參數的方式提供。僅限位置參數是在函數定義中出現在
/
之前(如果有)的條目。POSITIONAL_OR_KEYWORD
值既可以以關(guān)鍵字參數的形式提供,也可以以位置參數的形式提供(這是 Python 寫(xiě)成的函數的標準綁定行為的)。
VAR_POSITIONAL
沒(méi)有綁定到其他形參的位置實(shí)參組成的元組。這對應于 Python 函數定義中的
*args
形參。KEYWORD_ONLY
值必須以關(guān)鍵字參數的形式提供。僅限關(guān)鍵字形參是在 Python 函數定義中出現在
*
或*args
之后的條目。VAR_KEYWORD
一個(gè)未綁定到其他形參的關(guān)鍵字參數的字典。這對應于 Python 函數定義中的
**kwargs
形參。例如,打印所有沒(méi)有默認值的僅關(guān)鍵字參數:
>>> def foo(a, b, *, c, d=10): ... pass >>> sig = signature(foo) >>> for param in sig.parameters.values(): ... if (param.kind == param.KEYWORD_ONLY and ... param.default is param.empty): ... print('Parameter:', param) Parameter: c
- kind.description?
描述 Parameter.kind 的枚舉值。
3.8 新版功能.
例子:打印所有參數的描述:
>>> def foo(a, b, *, c, d=10): ... pass >>> sig = signature(foo) >>> for param in sig.parameters.values(): ... print(param.kind.description) positional or keyword positional or keyword keyword-only keyword-only
- replace(*[, name][, kind][, default][, annotation])?
基于 replace 函數被調用的目標,創(chuàng )建一個(gè)新的 Parameter 實(shí)例。要覆寫(xiě)
Parameter
的屬性,傳遞對應的參數。要移除 Parameter 的默認值和/或標注,傳遞Parameter.empty
。>>> from inspect import Parameter >>> param = Parameter('foo', Parameter.KEYWORD_ONLY, default=42) >>> str(param) 'foo=42' >>> str(param.replace()) # Will create a shallow copy of 'param' 'foo=42' >>> str(param.replace(default=Parameter.empty, annotation='spam')) "foo:'spam'"
在 3.4 版更改: 在 Python 3.3 中, Parameter 對象當
kind
為POSITIONAL_ONLY
時(shí)允許name
被設置為None
。這現在已不再被允許。
- class inspect.BoundArguments?
調用
Signature.bind()
或Signature.bind_partial()
的結果。容納實(shí)參到函數的形參的映射。- arguments?
一個(gè)形參名到實(shí)參值的可變映射。僅包含顯式綁定的參數。對
arguments
的修改會(huì )反映到args
和kwargs
上。應當在任何參數處理目的中與
Signature.parameters
結合使用。備注
Signature.bind()
和Signature.bind_partial()
中采用默認值的參數被跳過(guò)。然而,如果有需要的話(huà),可以使用BoundArguments.apply_defaults()
來(lái)添加它們。在 3.9 版更改:
arguments
現在的類(lèi)型是dict
。之前,它的類(lèi)型是collections.OrderedDict
。
- apply_defaults()?
設置缺失的參數的默認值。
對于變長(cháng)位置參數(
*args
),默認值是一個(gè)空元組。對于變長(cháng)關(guān)鍵字參數(
**kwargs
)默認值是一個(gè)空字典。>>> def foo(a, b='ham', *args): pass >>> ba = inspect.signature(foo).bind('spam') >>> ba.apply_defaults() >>> ba.arguments {'a': 'spam', 'b': 'ham', 'args': ()}
3.5 新版功能.
def test(a, *, b): ... sig = signature(test) ba = sig.bind(10, b=20) test(*ba.args, **ba.kwargs)
參見(jiàn)
- PEP 362 - 函數簽名對象。
包含具體的規范,實(shí)現細節和樣例。
類(lèi)與函數?
- inspect.getclasstree(classes, unique=False)?
將給定的類(lèi)的列表組織成嵌套列表的層級結構。每當一個(gè)內層列表出現時(shí),它包含的類(lèi)均派生自緊接著(zhù)該列表之前的條目的類(lèi)。每個(gè)條目均是一個(gè)二元組,包含一個(gè)類(lèi)和它的基類(lèi)組成的元組。如果 unique 參數為真值,則給定列表中的每個(gè)類(lèi)將恰有一個(gè)對應條目。否則,運用了多重繼承的類(lèi)和它們的后代將出現多次。
- inspect.getfullargspec(func)?
獲取一個(gè) Python 函數的形參的名字和默認值。將返回一個(gè) 具名元組:
FullArgSpec(args, varargs, varkw, defaults, kwonlyargs, kwonlydefaults, annotations)
args 是一個(gè)位置參數的名字的列表。 varargs 是
*
形參的名字或None
表示不接受任意長(cháng)位置參數時(shí)。 varkw 是**
參數的名字,或None
表示不接受任意關(guān)鍵字參數。 defaults 是一個(gè)包含了默認參數值的 n 元組分別對應最后 n 個(gè)位置參數,或None
則表示沒(méi)有默認值。 kwonlyargs 是一個(gè)僅關(guān)鍵詞參數列表,保持定義時(shí)的順序。 kwonlydefaults 是一個(gè)字典映射自 kwonlyargs 中包含的形參名。 annotations 是一個(gè)字典,包含形參值到標注的映射。其中包含一個(gè)特殊的鍵"return"
代表函數返回值的標注(如果有的話(huà))。注意:
signature()
和 Signature 對象 提供可調用對象內省更推薦的 API,并且支持擴展模塊 API 中可能出現的額外的行為(比如僅限位置參數)。該函數被保留的主要原因是保持兼容 Python 2 的inspect
模塊 API。在 3.4 版更改: 該函數現在基于
signature()
但仍然忽略__wrapped__
屬性,并且在簽名中包含綁定方法中的第一個(gè)綁定參數。在 3.6 版更改: 該方法在 Python 3.5 中曾因
signature()
被文檔歸為棄用。但該決定已被推翻以恢復一個(gè)明確受支持的標準接口,以便運用一份源碼通用 Python 2/3 間遺留的getargspec()
API 的遷移。在 3.7 版更改: Python 從 3.7 版起才顯式地保證了它保持僅關(guān)鍵字參數的定義順序,盡管實(shí)踐上在 Python 3 中一直保持了這個(gè)順序。
- inspect.getargvalues(frame)?
獲取傳入特定的幀的實(shí)參的信息。將返回一個(gè) 具名元組
ArgInfo(args, varargs, keywords, locals)
。 args 是一個(gè)參數名字的列表。 varargs 和 keyword 是*
和**
參數的名字或None
。 locals 是給定的幀的局部環(huán)境字典。備注
該函數因疏忽在 Python 3.5 中被錯誤地標記為棄用。
- inspect.formatargvalues(args[, varargs, varkw, locals, formatarg, formatvarargs, formatvarkw, formatvalue])?
將
getargvalues()
返回的四個(gè)值格式化為美觀(guān)的參數規格。 format* 的參數是對應的可選格式化函數以轉化名字和值為字符串。備注
該函數因疏忽在 Python 3.5 中被錯誤地標記為棄用。
- inspect.getmro(cls)?
返回由類(lèi) cls 的全部基類(lèi)按方法解析順序組成的元組,包括 cls 本身。所有類(lèi)不會(huì )在此元組中出現多于一次。注意方法解析順序取決于 cls 的類(lèi)型。除非使用一個(gè)非常奇怪的用戶(hù)定義元類(lèi)型,否則 cls 會(huì )是元組的第一個(gè)元素。
- inspect.getcallargs(func, /, *args, **kwds)?
仿照調用方式綁定 args 和 kwds 到 Python 函數或方法 func 參數名。對于綁定方法,也綁定第一個(gè)參數(通常命名為
self
)到關(guān)聯(lián)的實(shí)例。返回一個(gè)字典,映射自參數名(包括可能存在的*
和**
參數)到他們對應于 args 和 kwds 中的值。假使 func 被錯誤地調用,即是說(shuō)func(*args, **kwds)
會(huì )因函數簽名不一致引發(fā)一個(gè)異常,那么也會(huì )引發(fā)一個(gè)相同種類(lèi)的異常,并附上相同或類(lèi)似的消息。例如:>>> from inspect import getcallargs >>> def f(a, b=1, *pos, **named): ... pass >>> getcallargs(f, 1, 2, 3) == {'a': 1, 'named': {}, 'b': 2, 'pos': (3,)} True >>> getcallargs(f, a=2, x=4) == {'a': 2, 'named': {'x': 4}, 'b': 1, 'pos': ()} True >>> getcallargs(f) Traceback (most recent call last): ... TypeError: f() missing 1 required positional argument: 'a'
3.2 新版功能.
3.5 版后已移除: 改使用
Signature.bind()
和Signature.bind_partial()
。
- inspect.getclosurevars(func)?
獲取自 Python 函數或方法 func 引用的外部名字到它們的值的映射。返回一個(gè) 具名元組
ClosureVars(nonlocals, globals, builtins, unbound)
。 nonlocals 映射引用的名字到詞法閉包變量,globals 映射到函數的模塊級全局, builtins 映射到函數體內可見(jiàn)的內置變量。 unbound 是在函數中引用但不能解析到給定的模塊全局和內置變量的名字的集合。如果 func 不是 Python 函數或方法,將引發(fā)
TypeError
。3.3 新版功能.
- inspect.unwrap(func, *, stop=None)?
獲取 func 所包裝的對象。它追蹤
__wrapped__
屬性鏈并返回最后一個(gè)對象。stop 是一個(gè)可選的回調,接受包裝鏈的一個(gè)對象作為唯一參數,以允許通過(guò)讓回調返回真值使解包裝更早中止。如果回調不曾返回一個(gè)真值,將如常返回鏈中的最后一個(gè)對象。例如,
signature()
使用該參數來(lái)在遇到具有__signature__
參數的對象時(shí)停止解包裝。如果遇到循環(huán),則引發(fā)
ValueError
。3.4 新版功能.
- inspect.get_annotations(obj, *, globals=None, locals=None, eval_str=False)?
計算一個(gè)對象的標注字典。
obj
可以是一個(gè)可調用對象、類(lèi)或模塊。傳入其他類(lèi)型的對象將引發(fā)TypeError
。返回一個(gè)字典。
get_annotations()
每次調用均返回一個(gè)新的字典;對同一個(gè)對象調用兩次將獲得兩個(gè)不同但相等的字典。該函數幫助你處理若干細節:
如果
eval_str
為真,str
類(lèi)型的值將會(huì )通過(guò)eval()
來(lái)反字符串化。這被設計用于字符串化標注(from __future__ import annotations
)。如果
obj
不包含一個(gè)標注字典,返回一個(gè)空字典。(函數和方法永遠包含一個(gè)標注字典;類(lèi)、模塊和其他類(lèi)型的可調用對象則可能沒(méi)有。)對于類(lèi),忽略繼承而來(lái)的標注。如果一個(gè)類(lèi)不包含自己的標注字典,返回一個(gè)空字典。
因安全原因,所有對于對象成員和字典值的訪(fǎng)問(wèn)將通過(guò)
getattr()
和dict.get()
完成。永遠,永遠,永遠返回一個(gè)新建立的字典。
eval_str
控制str
類(lèi)型的值是否應該替換為對其調用eval()
的結果:如果 eval_str 為真,
eval()
會(huì )被調用于str
類(lèi)型。(注意get_annotations
并不捕獲異常;如果eval()
返回一個(gè)錯誤,它會(huì )將棧展開(kāi)跳過(guò)get_annotations
調用。)如果 eval_str 為假(默認值),
str
類(lèi)型的值將不會(huì )被改變。
globals
和locals
會(huì )被直接傳入函數eval()
,詳見(jiàn)eval()
的文檔。如果globals
或者locals
是None
,則改函數視type(obj)
而定,可能將相應的值替換為一個(gè)上下文有關(guān)的默認值。如果
obj
是一個(gè)模塊,globals
默認為obj.__dict__
。如果
obj
是一個(gè)類(lèi),globals
默認值為sys.modules[obj.__module__].__dict__
并且locals
默認值為obj
的類(lèi)命名空間。如果
obj
是一個(gè)可調用對象,globals
默認為obj.__globals__
,而如果obj
是一個(gè)包裝過(guò)的函數,則它會(huì )先被解包裝。
調用
get_annotations
是獲取任何對象的標注字典的最佳實(shí)踐。關(guān)于標注的最佳實(shí)踐的更多信息,參見(jiàn) 對象注解屬性的最佳實(shí)踐。3.10 新版功能.
解釋器棧?
Some of the following functions return
FrameInfo
objects. For backwards compatibility these objects allow
tuple-like operations on all attributes except positions
. This behavior
is considered deprecated and may be removed in the future.
- class inspect.FrameInfo?
- frame?
The frame object that the record corresponds to.
- filename?
The file name associated with the code being executed by the frame this record corresponds to.
- lineno?
The line number of the current line associated with the code being executed by the frame this record corresponds to.
- function?
The function name that is being executed by the frame this record corresponds to.
- code_context?
A list of lines of context from the source code that's being executed by the frame this record corresponds to.
- index?
The index of the current line being executed in the
code_context
list.
- positions?
A
dis.Positions
object containing the start line number, end line number, start column offset, and end column offset associated with the instruction being executed by the frame this record corresponds to.
在 3.5 版更改: 不再返回一個(gè)元組而是返回一個(gè)具名元組。
在 3.11 版更改: Changed the return object from a named tuple to a regular object (that is backwards compatible with the previous named tuple).
- class inspect.Traceback?
- filename?
The file name associated with the code being executed by the frame this traceback corresponds to.
- lineno?
The line number of the current line associated with the code being executed by the frame this traceback corresponds to.
- function?
The function name that is being executed by the frame this traceback corresponds to.
- code_context?
A list of lines of context from the source code that's being executed by the frame this traceback corresponds to.
- index?
The index of the current line being executed in the
code_context
list.
- positions?
A
dis.Positions
object containing the start line number, end line number, start column offset, and end column offset associated with the instruction being executed by the frame this traceback corresponds to.
備注
保留幀對象的引用(可見(jiàn)于這些函數返回的幀記錄的第一個(gè)元素)會(huì )導致你的程序產(chǎn)生循環(huán)引用。每當一個(gè)循環(huán)引用被創(chuàng )建,所有可從產(chǎn)生循環(huán)的對象訪(fǎng)問(wèn)的對象的生命周期將會(huì )被大幅度延長(cháng),即便 Python 的可選的循環(huán)檢測器被啟用。如果這類(lèi)循環(huán)必須被創(chuàng )建,確保它們會(huì )被顯式地打破以避免對象銷(xiāo)毀被延遲從而導致占用內存增加。
盡管循環(huán)檢測器能夠處理這種情況,這些幀(包括其局部變量)的銷(xiāo)毀可以通過(guò)在 finally
子句中移除循環(huán)來(lái)產(chǎn)生確定的行為。對于循環(huán)檢測器在編譯 Python 時(shí)被禁用或者使用 gc.disable()
時(shí),這樣處理更加尤為重要。比如:
def handle_stackframe_without_leak():
frame = inspect.currentframe()
try:
# do something with the frame
finally:
del frame
如果你希望保持幀更長(cháng)的時(shí)間(比如在之后打印回溯),你也可以通過(guò) frame.clear()
方法打破循環(huán)引用。
大部分這些函數支持的可選的 context 參數指定返回時(shí)包含的上下文的行數,以當前行為中心。
- inspect.getframeinfo(frame, context=1)?
Get information about a frame or traceback object. A
Traceback
object is returned.在 3.11 版更改: A
Traceback
object is returned instead of a named tuple.
- inspect.getouterframes(frame, context=1)?
Get a list of
FrameInfo
objects for a frame and all outer frames. These frames represent the calls that lead to the creation of frame. The first entry in the returned list represents frame; the last entry represents the outermost call on frame's stack.在 3.5 版更改: 返回一個(gè) 具名元組
FrameInfo(frame, filename, lineno, function, code_context, index)
的列表。在 3.11 版更改: A list of
FrameInfo
objects is returned.
- inspect.getinnerframes(traceback, context=1)?
Get a list of
FrameInfo
objects for a traceback's frame and all inner frames. These frames represent calls made as a consequence of frame. The first entry in the list represents traceback; the last entry represents where the exception was raised.在 3.5 版更改: 返回一個(gè) 具名元組
FrameInfo(frame, filename, lineno, function, code_context, index)
的列表。在 3.11 版更改: A list of
FrameInfo
objects is returned.
- inspect.currentframe()?
返回調用者的棧幀對應的幀對象。
CPython implementation detail: 該函數依賴(lài)于 Python 解釋器對于棧幀的支持,這并非在 Python 的所有實(shí)現中被保證。該函數在不支持Python 棧幀的實(shí)現中運行會(huì )返回
None
。
- inspect.stack(context=1)?
Return a list of
FrameInfo
objects for the caller's stack. The first entry in the returned list represents the caller; the last entry represents the outermost call on the stack.在 3.5 版更改: 返回一個(gè) 具名元組
FrameInfo(frame, filename, lineno, function, code_context, index)
的列表。在 3.11 版更改: A list of
FrameInfo
objects is returned.
- inspect.trace(context=1)?
Return a list of
FrameInfo
objects for the stack between the current frame and the frame in which an exception currently being handled was raised in. The first entry in the list represents the caller; the last entry represents where the exception was raised.在 3.5 版更改: 返回一個(gè) 具名元組
FrameInfo(frame, filename, lineno, function, code_context, index)
的列表。在 3.11 版更改: A list of
FrameInfo
objects is returned.
靜態(tài)地獲取屬性?
getattr()
和 hasattr()
都可能會(huì )在獲取或者判斷屬性是否存在時(shí)觸發(fā)代碼執行。描述符,就和特性一樣,會(huì )被調用, __getattr__()
和 __getattribute__()
可能會(huì )被調用。
對于你想要靜態(tài)地內省的情況,比如文檔工具,這會(huì )顯得不方便。 getattr_static()
擁有與 getattr()
相同的簽名,但避免了獲取屬性時(shí)執行代碼。
- inspect.getattr_static(obj, attr, default=None)?
獲取屬性而不觸發(fā)描述器協(xié)議的動(dòng)態(tài)查找能力
__getattr__()
或__getattribute__()
。注意:該函數可能無(wú)法獲取 getattr 能獲取的全部的屬性(比如動(dòng)態(tài)地創(chuàng )建的屬性),并且可能發(fā)現一些 getattr 無(wú)法找到的屬性(比如描述器會(huì )引發(fā) AttributeError)。它也能夠返回描述器對象本身而非實(shí)例成員。
如果實(shí)例的
__dict__
被其他成員遮蓋(比如一個(gè)特性)則該函數無(wú)法找到實(shí)例成員。3.2 新版功能.
getattr_static()
不解析描述器。比如槽描述器或 C 語(yǔ)言中實(shí)現的 getset 描述器。該描述器對象會(huì )被直接返回,而不處理底層屬性。
你可以用類(lèi)似下方的代碼的方法處理此事。注意,對于任意 getset 描述符,使用這段代碼仍可能觸發(fā)代碼執行。
# example code for resolving the builtin descriptor types
class _foo:
__slots__ = ['foo']
slot_descriptor = type(_foo.foo)
getset_descriptor = type(type(open(__file__)).name)
wrapper_descriptor = type(str.__dict__['__add__'])
descriptor_types = (slot_descriptor, getset_descriptor, wrapper_descriptor)
result = getattr_static(some_object, 'foo')
if type(result) in descriptor_types:
try:
result = result.__get__()
except AttributeError:
# descriptors can raise AttributeError to
# indicate there is no underlying value
# in which case the descriptor itself will
# have to do
pass
生成器和協(xié)程的當前狀態(tài)?
當實(shí)現協(xié)程調度器或其他更高級的生成器用途時(shí),判斷一個(gè)生成器是正在執行、等待啟動(dòng)或繼續或執行,又或者已經(jīng)被終止是非常有用的。 getgeneratorstate()
允許方便地判斷一個(gè)生成器的當前狀態(tài)。
- inspect.getgeneratorstate(generator)?
獲取生成器迭代器的當前狀態(tài)。
- 可能的狀態(tài)是:
GEN_CREATED:等待開(kāi)始執行。
GEN_RUNNING:正在被解釋器執行。
GEN_SUSPENDED:當前掛起于一個(gè) yield 表達式。
GEN_CLOSED:執行已經(jīng)完成。
3.2 新版功能.
- inspect.getcoroutinestate(coroutine)?
獲取協(xié)程對象的當前狀態(tài)。該函數設計為用于使用
async def
函數創(chuàng )建的協(xié)程函數,但也能接受任何包括cr_running
和cr_frame
的類(lèi)似協(xié)程的對象。- 可能的狀態(tài)是:
CORO_CREATED:等待開(kāi)始執行。
CORO_RUNNING:當前正在被解釋器執行。
CORO_SUSPENDED:當前掛起于一個(gè) await 表達式。
CORO_CLOSED:執行已經(jīng)完成。
3.5 新版功能.
生成器當前的內部狀態(tài)也可以被查詢(xún)。這通常在測試目的中最為有用,來(lái)保證內部狀態(tài)如預期一樣被更新:
- inspect.getgeneratorlocals(generator)?
獲取 generator 里的實(shí)時(shí)局部變量到當前值的映射。返回一個(gè)由名字映射到值的字典。這與在生成器的主體內調用
locals()
是等效的,并且相同的警告也適用。如果 generator 是一個(gè)沒(méi)有關(guān)聯(lián)幀的 生成器,則返回一個(gè)空字典。如果 generator 不是一個(gè) Python 生成器對象,則引發(fā)
TypeError
。CPython implementation detail: 該函數依賴(lài)于生成器為內省暴露一個(gè) Python 棧幀,這并非在 Python 的所有實(shí)現中被保證。在這種情況下,該函數將永遠返回一個(gè)空字典。
3.3 新版功能.
- inspect.getcoroutinelocals(coroutine)?
該函數可類(lèi)比于
getgeneratorlocals()
,只是作用于由async def
函數創(chuàng )建的協(xié)程。3.5 新版功能.
代碼對象位標志?
Python 代碼對象有一個(gè) co_flags
屬性,它是下列標志的位圖。
- inspect.CO_OPTIMIZED?
代碼對象已經(jīng)經(jīng)過(guò)優(yōu)化,會(huì )采用快速局部變量。
- inspect.CO_NEWLOCALS?
如果被置位,當代碼對象被執行時(shí)會(huì )創(chuàng )建一個(gè)新的字典作為幀的
f_locals
。
- inspect.CO_VARARGS?
代碼對象擁有一個(gè)變長(cháng)位置形參(類(lèi)似
*args
)。
- inspect.CO_VARKEYWORDS?
代碼對象擁有一個(gè)可變關(guān)鍵字形慘死(類(lèi)似
**kwrags
)。
- inspect.CO_NESTED?
該標志當代碼對象是一個(gè)嵌套函數時(shí)被置位。
- inspect.CO_GENERATOR?
當代碼對象是一個(gè)生成器函數,即調用時(shí)會(huì )返回一個(gè)生成器對象,則該標志被置位。
- inspect.CO_COROUTINE?
當代碼對象是一個(gè)協(xié)程函數時(shí)被置位。當代碼對象被執行時(shí)它返回一個(gè)協(xié)程。詳見(jiàn) PEP 492。
3.5 新版功能.
- inspect.CO_ITERABLE_COROUTINE?
該標志被用于將生成器轉變?yōu)榛谏善鞯膮f(xié)程。包含此標志的生成器對象可以被用于
await
表達式,并可以yield from
協(xié)程對象。詳見(jiàn) PEP 492。3.5 新版功能.
- inspect.CO_ASYNC_GENERATOR?
當代碼對象是一個(gè)異步生成器函數時(shí)該標志被置位。當代碼對象被運行時(shí)它將返回一個(gè)異步生成器對象。詳見(jiàn) PEP 525。
3.6 新版功能.
備注
這些標志特指于 CPython,并且在其他 Python 實(shí)現中可能從未被定義。更進(jìn)一步地說(shuō),這些標志是一種實(shí)現細節,并且可能在將來(lái)的 Python 發(fā)行中被移除或棄用。推薦使用 inspect
模塊的公共 API 來(lái)進(jìn)行任何內省需求。
命令行界面?
inspect
模塊也提供一個(gè)從命令行使用基本的內省能力。
默認地,命令行接受一個(gè)模塊的名字并打印模塊的源代碼。也可通過(guò)后綴一個(gè)冒號和目標對象的限定名稱(chēng)來(lái)打印一個(gè)類(lèi)或者一個(gè)函數。
- --details?
打印特定對象的信息而非源碼。