Py tricks(1): python實現不可修改的常量

由於種種緣由,Python並未提供如C/C++/Java同樣的const修飾符,換言之,python中沒有常量,至少截止2015年年底,尚未這個打算。Python程序通常經過約定俗成的變量名全大寫的形式來表示這是一個常量,可是這終究不是長久之計。python

其實Python能夠曲線救國實現常量。函數

在Python的面向對象中,object.__setattr__()這個built-in function在對類的屬性賦值的時候會自動調用。其函數原型爲:測試

object.__setattr__(self, name, value)

其中name爲變量名,value爲變量值。ui

object.__dict__則以dict的形式保存了object內全部可寫的屬性,key爲變量名,value爲變量值。code

那麼咱們就有可能經過創建一個const類,對其object.__setattr__()方法進行overwrite,在對屬性值進行賦值的時候判斷,若是屬性存在,則表示這是對常量的重賦值操做,從而拋出異常,若是屬性不存在,則表示是新聲明瞭一個常量,能夠進行賦值操做。對象

const.py 代碼以下:utf-8

# -*- 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,然而咱們但願用更簡潔的方法進行常量的賦值。形如:it

import const

const.TEST_CONSTANT = 'test'

在python中,__name__內置屬性是當前的class或者type的值。通俗地講,__name__的值有如下兩種形式:io

  • 若是運行某一個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

打印信息以下:

test
test1
Traceback (most recent call last):
  File "H:/code/test.py", line 9, in <module>
    const.TEST = 'test'
  File "H:\code\const.py", line 9, in __setattr__
    raise self.ConstError,"constant reassigning error!"
const.ConstError: constant reassignment error!

成功爲兩常量賦值,在試圖修改第一個常量值時拋出異常:)

相關文章
相關標籤/搜索