粘貼自:https://www.cnblogs.com/Vito2008/p/5006255.html
操作步驟:在一個.py文件中寫好一個const的文件。然后將這個文件放在安裝python的lib路徑下(具體const文件如截圖)
python實現不可修改的常量
因為種種原因,Python並未提供如C/C++/Java一樣的const修飾符,換言之,python中沒有常量,至少截止2015年年末,還沒有這個打算。Python程序一般通過約定俗成的變量名全大寫的形式來表示這是一個常量,但是這終究不是長久之計。
其實Python可以曲線救國實現常量。
在Python的面向對象中,
object.setattr()
這個built-in function在對類的屬性賦值的時候會自動調用。其函數原型為:
object.setattr(self, name, value)
其中name為變量名,value為變量值。
而object.__dict__則以dict的形式保存了object內所有可寫的屬性,key為變量名,value為變量值。
那么我們就有可能通過建立一個const類,對其object.setattr()方法進行overwrite,在對屬性值進行賦值的時候判斷,如果屬性存在,則表示這是對常量的重賦值操作,從而拋出異常,如果屬性不存在,則表示是新聲明了一個常量,可以進行賦值操作。
const.py 代碼如下:
復制代碼
-- coding: utf-8 --
class _const:
class ConstError(TypeError) : pass
def setattr(self, key, value):
# self.dict
if self.dict.has_key(key):
raise self.ConstError,"constant reassignment error!"
self.dict[key] = value
import sys
sys.modules[name] = _const()
復制代碼
其中,1-10行是上述思路的類的一個實現。
第12-14行的寫法值得說明。我們盡管擁有了_const類,但是我們當前使用這個類仍然需要
import const
c = const._const()
c.TEST_CONSTANT = 'test'
這樣的形式來聲明一個常量TEST_CONSTANT,然而我們希望用更簡潔的方法進行常量的賦值。形如:
import const
const.TEST_CONSTANT = 'test'
在python中,__name__內置屬性是當前的class或者type的值。通俗地講,__name__的值有以下兩種形式:
如果運行某一個py文件,在該文件中,__name__的值為'main'
如果import了某一個py文件,那么在該import的文件中,__name__的值為該文件的文件名(不帶.py后綴)
而sys.modules是一個dict對象,包括了當前上下文中python已經load的所有模塊的信息,dict的key為文件名,value為模塊對象。
在const.py 中,14行的寫法等價於
import const
sys.modules['const'] = _const()
即,讓_const類作為模塊的入口點,引入const.py等價於聲明了一個_const類的實例。
至此python的常量實現完畢,使用test.py測試:
復制代碼
-- coding: utf-8 --
import const
const.TEST = 'test'
print const.TEST
const.TEST1 = 'test1'
print const.TEST1
const.TEST = 'test'
print const.TEST
復制代碼
打印信息如下:
1
2
3
4
5
6
7
8
test
test1
Traceback (most recent call last):
File "H:/code/test.py", line 9, in
const.TEST = 'test'
File "H:\code\const.py", line 9, in
setattr
raise self.ConstError,"constant reassigning error!"
const.ConstError: constant reassignment error!
成功為兩常量賦值,在試圖修改第一個常量值時拋出異常:)
標簽: Python