Python?eval()和exec()函數(shù)使用詳解
eval() 和 exec() 函數(shù)都屬于 Python 的內(nèi)置函數(shù)。
eval() 和 exec() 函數(shù)的功能是相似的,都可以執(zhí)行一個字符串形式的 Python 代碼(代碼以字符串的形式提供),相當(dāng)于一個 Python 的解釋器。二者不同之處在于,eval() 執(zhí)行完要返回結(jié)果,而 exec() 執(zhí)行完不返回結(jié)果。
eval()和exec()的用法
eval() 函數(shù)的語法格式為:
eval(expression, globals=None, locals=None, /)
而 exec() 函數(shù)的語法格式如下:
exec(expression, globals=None, locals=None, /)
可以看到,二者的語法格式除了函數(shù)名,其他都相同,其中各個參數(shù)的具體含義如下:
1、expression:這個參數(shù)是一個字符串,代表要執(zhí)行的語句 。該語句受后面兩個字典類型參數(shù) globals 和 locals 的限制,只有在 globals 字典和 locals 字典作用域內(nèi)的函數(shù)和變量才能被執(zhí)行。
2、globals:這個參數(shù)管控的是一個全局的命名空間,即 expression 可以使用全局命名空間中的函數(shù)。如果只是提供了 globals 參數(shù),而沒有提供自定義的 builtins,則系統(tǒng)會將當(dāng)前環(huán)境中的 builtins 復(fù)制到自己提供的 globals 中,然后才會進(jìn)行計算;如果連 globals 這個參數(shù)都沒有被提供,則使用 Python 的全局命名空間。
3、locals:這個參數(shù)管控的是一個局部的命名空間,和 globals 類似,當(dāng)它和 globals 中有重復(fù)或沖突時,以 locals 的為準(zhǔn)。如果 locals 沒有被提供,則默認(rèn)為 globals。
它們的區(qū)別在于,eval() 執(zhí)行完會返回結(jié)果,而 exec() 執(zhí)行完不返回結(jié)果
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)更易擴(kuò)展。
TensorFlow 框架,就會發(fā)現(xiàn)該框架中的靜態(tài)圖就是類似這個原理實現(xiàn)的:
TensorFlow 中先將張量定義在一個靜態(tài)圖里,這就相當(dāng)將鍵值對添加到字典里一樣; TensorFlow 中通過 session 和張量的 eval() 函數(shù)來進(jìn)行具體值的運(yùn)算,就當(dāng)于使用 eval() 函數(shù)進(jìn)行具體值的運(yùn)算一樣。
需要注意的是,在使用 eval() 或是 exec() 來處理請求代碼時,函數(shù) eval() 和 exec() 常常會被黑客利用,成為可以執(zhí)行系統(tǒng)級命令的入口點,進(jìn)而來攻擊網(wǎng)站。解決方法是:通過設(shè)置其命名空間里的可執(zhí)行函數(shù),來限制 eval() 和 exec() 的執(zhí)行范圍。
python中的exec()函數(shù)和eval()函數(shù)
exec()函數(shù)
exec函數(shù)用于執(zhí)行存儲在字符串中的python語句
>>> exec("x=1") >>> x 1
但有時候,直接這樣執(zhí)行可能會污染我們的命名空間,比如下面的例子,我們使用python內(nèi)置的abs函數(shù)求絕對值。
>>> abs(-1) 1 >>> exec("abs='xyz'") >>> abs(-1) File "<stdin>", line 1, in <module> TypeError: "str" object is not callable
在執(zhí)行abs函數(shù)時報錯了,類型錯誤,字符串類型對象不能被調(diào)用,原因就在于我們使用exec函數(shù)將字符串xyz賦值給了abs,abs此時已經(jīng)不再表示求絕對值的函數(shù)了。為了防止出現(xiàn)這種情況,我們在調(diào)用exec函數(shù)時,可以給它傳遞第二個參數(shù)——命名空間,即就是一個字典。
>>> abs(-1) 1 >>> scope = {} >>> exec("abs='xyz'", scope) >>> abs(-1) 1 >>>scope['abs'] 'xyz'
這樣,通過exec執(zhí)行賦值語句創(chuàng)建的變量就位于scope中,不會污染我們的命名空間。
eval()函數(shù)
eval用于執(zhí)行存儲在字符串中的python表達(dá)式
>>> eval("1+2+3+4+5") 15
與exec函數(shù)一樣,我們也可以給eval函數(shù)提供命名空間。
總結(jié)
1、exec函數(shù)執(zhí)行的是python語句,沒有返回值,eval函數(shù)執(zhí)行的是python表達(dá)式,有返回值;
2、exec函數(shù)和eval函數(shù)都可以傳入命名空間作為參數(shù),實際上,可以向exec函數(shù)和eval函數(shù)提供兩個命名空間,他們的函數(shù)定義為:
exec(source, globals=None, locals=None)
eval(source, globals=None, locals=None)
其中g(shù)lobals和locals都是可選參數(shù),globals表示全局命名空間,必須是字典,locals表示局部命名空間,可以是任何映射。
3、需要注意的是,exec函數(shù)和eval函數(shù)都是將用戶提供的字符串作為代碼執(zhí)行,將無法控制代碼的行為,會帶來嚴(yán)重的安全隱患,使用的時候要慎重。
到此這篇關(guān)于Python eval()和exec()函數(shù)的文章就介紹到這了,更多相關(guān)Python eval()和exec()函數(shù)內(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函數(shù)默認(rèn)參數(shù)使用避坑指南
這篇文章主要為大家介紹了python函數(shù)默認(rèn)參數(shù)使用的踩雷避坑詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-07-07pytorch張量和numpy數(shù)組相互轉(zhuǎn)換
在使用pytorch作為深度學(xué)習(xí)的框架時,經(jīng)常會遇到張量tensor和矩陣numpy的類型的相互轉(zhuǎn)化的問題,本文主要介紹了pytorch張量和numpy數(shù)組相互轉(zhuǎn)換,感興趣的可以了解一下2024-02-02python和C++共享內(nèi)存?zhèn)鬏攬D像的示例
這篇文章主要介紹了python和C++共享內(nèi)存?zhèn)鬏攬D像的示例,幫助大家利用python處理圖片,感興趣的朋友可以了解下2020-10-10