由于种种缘由,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!
成功为两常量赋值,在试图修改第一个常量值时抛出异常:)