python實(shí)現(xiàn)可變變量名方法詳解
如果要寫一個(gè)程序,讓x1為1,x2為2,然后直到x100為100,你會(huì)怎么做?
在C這種靜態(tài)語言里,變量名這個(gè)標(biāo)識(shí)符實(shí)際上會(huì)被編譯器直接翻譯成內(nèi)存地址,所以除了手動(dòng)設(shè)置每個(gè)變量的值以外,沒辦法做到這點(diǎn)。而Python這種動(dòng)態(tài)語言則是可以做到的。
最容易想到的自然是eval,但是實(shí)際上根本不需要這種危險(xiǎn)的東西,因?yàn)镻ython的變量名就是一個(gè)字典的key而已。要獲取這個(gè)字典,直接用locals和globals函數(shù)即可。
因此這個(gè)程序可以這樣實(shí)現(xiàn):
代碼如下:
>>> names = locals() >>> for i in xrange(1, 101): ... names['x%s' % i] = i ... >>> x1 1 >>> x2 2 >>> x100 100
不過你也許會(huì)說這個(gè)例子沒什么用,畢竟用數(shù)組來實(shí)現(xiàn)更為實(shí)用。
那么再考慮一個(gè)例子:服務(wù)器使用一種對(duì)象數(shù)據(jù)庫,可以直接保存對(duì)象到數(shù)據(jù)庫中。服務(wù)器列出目前支持的所有類,而用戶想添加一個(gè)不存在于列表中的類,于是向服務(wù)器發(fā)送一段JSON或XML文本。服務(wù)器解析這段文本,將它轉(zhuǎn)換成一個(gè)class對(duì)象,并且設(shè)置類名。之后用戶就可以隨意生成這個(gè)類的對(duì)象。
關(guān)鍵是這個(gè)數(shù)據(jù)庫和類名相關(guān),你不能用一個(gè)通用的Object類來保存所有的對(duì)象,否則查詢時(shí)就亂套了。
而恰巧的是,還就有人在GAE論壇上提出了這個(gè)需求,而只會(huì)Java的他最終只能放棄。
當(dāng)然,你想用來惡搞也行:
代碼如下:
>>> locals()['True'] = False >>> True False
另一個(gè)用處就是測(cè)試一個(gè)變量名是否已經(jīng)存在。標(biāo)準(zhǔn)的做法是try...except一個(gè)NameError異常,實(shí)際上直接用in locals()或in globals()就能判斷了。
順便再介紹另一種奇怪的方法,不知道有人這樣寫過沒:
代碼如下:
>>> import __main__ >>> hasattr(__main__, 'x') False >>> setattr(__main__, 'x', 1) >>> x 1 >>> hasattr(__main__, 'x') True
當(dāng)然,沒有任何人推薦你這樣寫,我也不會(huì)。
最后,除了動(dòng)態(tài)設(shè)置變量名,動(dòng)態(tài)刪除也是可以的,例如del locals()['x1']。同樣,delattr也是可用的。
知識(shí)點(diǎn)擴(kuò)展:
python 動(dòng)態(tài)獲取變量的變量名
需求目標(biāo):如果有了上面的動(dòng)態(tài)命名list,那么當(dāng)動(dòng)態(tài)獲取變量的變量名,就需要如下的操作:
利用python原生的inspect庫來實(shí)現(xiàn):
核心代碼:
import inspect def get_variable_name(variable): callers_local_vars = inspect.currentframe().f_back.f_locals.items() return [var_name for var_name, var_val in callers_local_vars if var_val is variable]
測(cè)試代碼:
def get_variable_name(variable): callers_local_vars = inspect.currentframe().f_back.f_locals.items() return [var_name for var_name, var_val in callers_local_vars if var_val is variable] if __name__ == '__main__': prepare_list = locals() for i in range(16): prepare_list['list_' + str(i)] = [] prepare_list['list_' + str(i)].append(('我是第' + str(i)) + '個(gè)list') a = get_variable_name(prepare_list['list_0']).pop() b = get_variable_name(prepare_list['list_1']).pop() print(a) print(b)
相關(guān)文章
python處理json字符串(使用json.loads而不是eval())
eval 跟json.loads 是不一樣的函數(shù),是有實(shí)現(xiàn)不一樣功能的地方,但是在某些地方它們兩個(gè)函數(shù)的功能是一樣的,本文就詳細(xì)介紹一下2021-09-09Python遞歸函數(shù)反轉(zhuǎn)序列的實(shí)現(xiàn)
本文主要介紹了Python遞歸函數(shù)反轉(zhuǎn)序列的實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2023-07-07python3.7將代碼打包成exe程序并添加圖標(biāo)的方法
這篇文章主要介紹了python3.7將代碼打包成exe程序并添加圖標(biāo)的方法,本文給大家介紹的非常詳細(xì),具有一定的參考借鑒價(jià)值 ,需要的朋友可以參考下2019-10-10python使用Windows的wmic命令監(jiān)控文件運(yùn)行狀況,如有異常發(fā)送郵件報(bào)警
這篇文章主要介紹了python使用Windows的wmic命令監(jiān)控文件運(yùn)行狀況,如有異常發(fā)送郵件報(bào)警的示例,幫助大家更好的理解和使用python,感興趣的朋友可以了解下2021-01-01