Py tricks(1): python实现不可修改的常量
admin
2023-07-31 01:43:32
0

因为种种原因,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

打印信息如下:

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!

成功为两常量赋值,在试图修改第一个常量值时抛出异常:)

相关内容

热门资讯

500 行 Python 代码... 语法分析器描述了一个句子的语法结构,用来帮助其他的应用进行推理。自然语言引入了很多意外的歧义,以我们...
定时清理删除C:\Progra... C:\Program Files (x86)下面很多scoped_dir开头的文件夹 写个批处理 定...
65536是2的几次方 计算2... 65536是2的16次方:65536=2⁶ 65536是256的2次方:65536=256 6553...
Mobi、epub格式电子书如... 在wps里全局设置里有一个文件关联,打开,勾选电子书文件选项就可以了。
scoped_dir32_70... 一台虚拟机C盘总是莫名奇妙的空间用完,导致很多软件没法再运行。经过仔细检查发现是C:\Program...
pycparser 是一个用... `pycparser` 是一个用 Python 编写的 C 语言解析器。它可以用来解析 C 代码并构...
小程序支付时提示:appid和... [Q]小程序支付时提示:appid和mch_id不匹配 [A]小程序和微信支付没有进行关联,访问“小...
Prometheus+Graf... 一,Prometheus概述 1,什么是Prometheus?Prometheus是最初在Sound...
python绘图库Matplo... 本文简单介绍了Python绘图库Matplotlib的安装,简介如下: matplotlib是pyt...
微信小程序使用slider实现... 众所周知哈,微信小程序里面的音频播放是没有进度条的,但最近有个项目呢,客户要求音频要有进度条控制,所...