Python eval()與exec()函數(shù)使用介紹
eval() 和 exec() 函數(shù)都屬于 Python 的內(nèi)置函數(shù),由于這兩個函數(shù)在功能和用法方面都有相似之處,所以將它們放到一節(jié)進行介紹。
eval() 和 exec() 函數(shù)的功能是相似的,都可以執(zhí)行一個字符串形式的 Python 代碼(代碼以字符串的形式提供),相當(dāng)于一個 Python 的解釋器。二者不同之處在于,eval() 執(zhí)行完要返回結(jié)果,而 exec() 執(zhí)行完不返回結(jié)果(文章后續(xù)會給出詳細(xì)示例)。
eval()和exec()的用法
eval() 函數(shù)的語法格式為:
eval(expression, globals=None, locals=None, /)
而 exec() 函數(shù)的語法格式如下:
exec(expression, globals=None, locals=None, /)
可以看到,二者的語法格式除了函數(shù)名,其他都相同,其中各個參數(shù)的具體含義如下:
- expression:這個參數(shù)是一個字符串,代表要執(zhí)行的語句 。該語句受后面兩個字典類型參數(shù) globals 和 locals 的限制,只有在 globals 字典和 locals 字典作用域內(nèi)的函數(shù)和變量才能被執(zhí)行。
- globals:這個參數(shù)管控的是一個全局的命名空間,即 expression 可以使用全局命名空間中的函數(shù)。如果只是提供了 globals 參數(shù),而沒有提供自定義的 __builtins__,則系統(tǒng)會將當(dāng)前環(huán)境中的 __builtins__ 復(fù)制到自己提供的 globals 中,然后才會進行計算;如果連 globals 這個參數(shù)都沒有被提供,則使用 Python 的全局命名空間。
- locals:這個參數(shù)管控的是一個局部的命名空間,和 globals 類似,當(dāng)它和 globals 中有重復(fù)或沖突時,以 locals 的為準(zhǔn)。如果 locals 沒有被提供,則默認(rèn)為 globals。
注意,__builtins__ 是 Python 的內(nèi)建模塊,平時使用的 int、str、abs 都在這個模塊中。通過 print(dic["__builtins__"]) 語句可以查看 __builtins__ 所對應(yīng)的 value。
首先,通過如下的例子來演示參數(shù) globals 作用域的作用,注意觀察它是何時將 __builtins__ 復(fù)制 globals 字典中去的:
dic={}#定義一個字典 dic['b']=3#在 dic 中加一條元素,key 為 b print(dic.keys())#先將 dic 的 key 打印出來,有一個元素 b exec("a = 4", dic)#在 exec 執(zhí)行的語句后面跟一個作用域 dic print(dic.keys())#exec 后,dic 的 key 多了一個
運行結(jié)果為:
dict_keys(['b'])
dict_keys(['b', '__builtins__', 'a'])
上面的代碼是在作用域 dic 下執(zhí)行了一句 a = 4 的代碼。可以看出,exec() 之前 dic 中的 key 只有一個 b。執(zhí)行完 exec() 之后,系統(tǒng)在 dic 中生成了兩個新的 key,分別是 a 和 __builtins__。其中,a 為執(zhí)行語句生成的變量,系統(tǒng)將其放到指定的作用域字典里;__builtins__ 是系統(tǒng)加入的內(nèi)置 key。
locals參數(shù)的用法就很簡單了,舉個例子:
a=10 b=20 c=30 g={'a':6,'b':8}#定義一個字典 t={'b':100,'c':10}#定義一個字典 print(eval('a+b+c', g, t))#定義一個字典 116
輸出結(jié)果為:
116
exec()和eval()的區(qū)別
前面已經(jīng)講過,它們的區(qū)別在于,eval() 執(zhí)行完會返回結(jié)果,而 exec() 執(zhí)行完不返回結(jié)果。舉個例子:
a =1 exec("a = 2")#相當(dāng)于直接執(zhí)行 a=2 print(a) a =exec("2+3")#相當(dāng)于直接執(zhí)行 2+3,但是并沒有返回值,a 應(yīng)為 None print(a) a =eval('2+3')#執(zhí)行 2+3,并把結(jié)果返回給 a print(a)
運行結(jié)果為:
2
None
5
可以看出,exec() 中最適合放置運行后沒有結(jié)果的語句,而 eval() 中適合放置有結(jié)果返回的語句。
如果 eval() 里放置一個沒有結(jié)果返回的語句會怎樣呢?例如下面代碼:
a=eval("a = 2")
這時 Python 解釋器會報 SyntaxError 錯誤,提示 eval() 中不識別等號語法。
eval() 和 exec() 函數(shù)的應(yīng)用場景
在使用 Python 開發(fā)服務(wù)端程序時,這兩個函數(shù)應(yīng)用得非常廣泛。例如,客戶端向服務(wù)端發(fā)送一段字符串代碼,服務(wù)端無需關(guān)心具體的內(nèi)容,直接跳過 eval() 或 exec() 來執(zhí)行,這樣的設(shè)計會使服務(wù)端與客戶端的耦合度更低,系統(tǒng)更易擴展。
另外,如果讀者以后接觸 TensorFlow 框架,就會發(fā)現(xiàn)該框架中的靜態(tài)圖就是類似這個原理實現(xiàn)的:
- TensorFlow 中先將張量定義在一個靜態(tài)圖里,這就相當(dāng)將鍵值對添加到字典里一樣;
- TensorFlow 中通過 session 和張量的 eval() 函數(shù)來進行具體值的運算,就當(dāng)于使用 eval() 函數(shù)進行具體值的運算一樣。
需要注意的是,在使用 eval() 或是 exec() 來處理請求代碼時,函數(shù) eval() 和 exec() 常常會被黑客利用,成為可以執(zhí)行系統(tǒng)級命令的入口點,進而來攻擊網(wǎng)站。解決方法是:通過設(shè)置其命名空間里的可執(zhí)行函數(shù),來限制 eval() 和 exec() 的執(zhí)行范圍。
到此這篇關(guān)于Python eval()與exec()函數(shù)使用介紹的文章就介紹到這了,更多相關(guān)Python eval()與exec()內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
- python中eval的用法及說明
- Python 中eval()函數(shù)的正確使用及其風(fēng)險分析(使用示例)
- python中eval函數(shù)使用與異常處理詳解
- Python使用eval函數(shù)解析和執(zhí)行字符串
- Python中的eval()函數(shù)使用詳解
- python中的exec()、eval()及complie()示例詳解
- Python中eval()函數(shù)的功能及使用方法小結(jié)
- python中關(guān)于eval函數(shù)的使用及說明
- Python?eval()和exec()函數(shù)使用詳解
- Python eval函數(shù)的實現(xiàn)
相關(guān)文章
利于python腳本編寫可視化nmap和masscan的方法
這篇文章主要介紹了利于python腳本編寫可視化nmap和masscan的方法,本文給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2020-12-12Python處理字節(jié)串:struct.pack和struct.unpack使用
這篇文章主要介紹了Python處理字節(jié)串:struct.pack和struct.unpack使用方式,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教2024-01-01Python DataFrame使用drop_duplicates()函數(shù)去重(保留重復(fù)值,取重復(fù)值)
這篇文章主要介紹了Python DataFrame使用drop_duplicates()函數(shù)去重(保留重復(fù)值,取重復(fù)值),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-07-07python人工智能深度學(xué)習(xí)算法優(yōu)化
這篇文章主要為大家介紹了python人工智能深度學(xué)習(xí)關(guān)于算法優(yōu)化詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步2021-11-11