Python 定义只读属性的实现方式


Posted in Python onMarch 05, 2020

Python是面向对象(OOP)的语言, 而且在OOP这条路上比Java走得更彻底, 因为在Python里, 一切皆对象, 包括int, float等基本数据类型.

在Java里, 若要为一个类定义只读的属性, 只需要将目标属性用private修饰, 然后只提供getter()而不提供setter(). 但Python没有private关键字, 如何定义只读属性呢? 有两种方法, 第一种跟Java类似, 通过定义私有属性实现. 第二种是通过__setattr__.

通过私有属性

Python里定义私有属性的方法见 https://3water.com/article/181953.htm.

用私有属性+@property定义只读属性, 需要预先定义好属性名, 然后实现对应的getter方法.

class Vector2D(object):
 def __init__(self, x, y):
 self.__x = float(x)
 self.__y = float(y)

 @property
 def x(self):
 return self.__x
 @property
 def y(self):
 return self.__y

if __name__ == "__main__":
 v = Vector2D(3, 4)
 print(v.x, v.y)
 v.x = 8 # error will be raised.

输出:

(3.0, 4.0)
Traceback (most recent call last):
 File ...., line 16, in <module>
 v.x = 8 # error will be raised.
AttributeError: can't set attribute

可以看出, 属性x是可读但不可写的.

通过__setattr__

当我们调用obj.attr=value时发生了什么?

很简单, 调用了obj的__setattr__方法. 可通过以下代码验证:

class MyCls():
 def __init__(self):
 pass

 def __setattr__(self, f, v):
 print 'setting %r = %r'%(f, v)
if __name__ == '__main__':
 obj = MyCls()
 obj.new_field = 1

输出:

setting 'new_field' = 1

所以呢, 只需要在__setattr__ 方法里挡一下, 就可以阻止属性值的设置, 可谓是釜底抽薪.

代码:

# encoding=utf8
class MyCls(object):
 readonly_property = 'readonly_property' 
 def __init__(self):
 pass
 def __setattr__(self, f, v):
 if f == 'readonly_property':
  raise AttributeError('{}.{} is READ ONLY'.\
     format(type(self).__name__, f))

 else:
  self.__dict__[f] = v

if __name__ == '__main__':
 obj = MyCls()

 obj.any_other_property = 'any_other_property'
 print(obj.any_other_property)

 print(obj.readonly_property)
 obj.readonly_property = 1

输出:

any_other_property
readonly_property
Traceback (most recent call last):
 File "...", line 21, in <module>
 obj.readonly_property = 1
 ...
 AttributeError: MyCls.readonly_property is READ ONLY

以上这篇Python 定义只读属性的实现方式就是小编分享给大家的全部内容了,希望能给大家一个参考,也希望大家多多支持三水点靠木。

Python 相关文章推荐
Python操作sqlite3快速、安全插入数据(防注入)的实例
Apr 26 Python
跟老齐学Python之坑爹的字符编码
Sep 28 Python
Python实用日期时间处理方法汇总
May 09 Python
python3序列化与反序列化用法实例
May 26 Python
Ubuntu下安装PyV8
Mar 13 Python
对python中的six.moves模块的下载函数urlretrieve详解
Dec 19 Python
selenium+python自动化测试之页面元素定位
Jan 23 Python
Python如何处理大数据?3个技巧效率提升攻略(推荐)
Apr 15 Python
Python 将json序列化后的字符串转换成字典(推荐)
Jan 06 Python
pygame实现飞机大战
Mar 11 Python
Python列表嵌套常见坑点及解决方案
Sep 30 Python
python调用win32接口进行截图的示例
Nov 11 Python
Pycharm中import torch报错的快速解决方法
Mar 05 #Python
Python中私有属性的定义方式
Mar 05 #Python
Python实现AI自动抠图实例解析
Mar 05 #Python
python GUI库图形界面开发之PyQt5 MDI(多文档窗口)QMidArea详细使用方法与实例
Mar 05 #Python
Python matplotlib修改默认字体的操作
Mar 05 #Python
Django 返回json数据的实现示例
Mar 05 #Python
python利用opencv实现SIFT特征提取与匹配
Mar 05 #Python
You might like
10条PHP高级技巧[修正版]
2011/08/02 PHP
PHP应用JSON技巧讲解
2013/02/03 PHP
ThinkPHP中的关联模型注意点
2014/06/16 PHP
php mysql数据库操作类(实例讲解)
2017/08/06 PHP
PHP如何实现阿里云短信sdk灵活应用在项目中的方法
2019/06/14 PHP
会自动逐行上升的文本框
2006/06/30 Javascript
textContent在Firefox下与innerText等效的属性
2007/05/12 Javascript
DHTML Slide Show script图片轮换
2008/03/03 Javascript
文本框的字数限制功能jquery插件
2009/11/24 Javascript
jquery禁止输入数字以外的字符的示例(纯数字验证码)
2014/04/10 Javascript
详解JS去重及字符串奇数位小写转大写
2016/12/29 Javascript
BootStrap Datepicker 插件修改为默认中文的实现方法
2017/02/10 Javascript
分享ES6的7个实用技巧
2018/01/18 Javascript
jQuery niceScroll滚动条错位问题的解决方法
2018/02/03 jQuery
layer弹出层显示在top顶层的方法
2019/09/11 Javascript
javascript网页随机点名实现过程解析
2019/10/15 Javascript
JS 数组基本用法入门示例解析
2020/01/16 Javascript
JS控制下拉列表左右选择实例代码
2020/05/08 Javascript
vue使用exif获取图片经纬度的示例代码
2020/12/11 Vue.js
[02:28]DOTA2亚洲邀请赛 LGD战队巡礼
2015/02/03 DOTA
Python中list初始化方法示例
2016/09/18 Python
pandas 实现字典转换成DataFrame的方法
2018/07/04 Python
python调用webservice接口的实现
2019/07/12 Python
浅谈Python_Openpyxl使用(最全总结)
2019/09/05 Python
flask的orm框架SQLAlchemy查询实现解析
2019/12/12 Python
用openCV和Python 实现图片对比,并标识出不同点的方式
2019/12/19 Python
Python爬虫小例子——爬取51job发布的工作职位
2020/07/10 Python
解决CSS3的opacity属性带来的层叠顺序问题
2016/05/09 HTML / CSS
英国健身仓库:Bodybuilding Warehouse
2019/03/06 全球购物
外企求职信范文分享
2013/12/31 职场文书
保证书格式范文
2014/04/28 职场文书
工作收入证明范本
2015/06/12 职场文书
浅谈如何写好演讲稿?
2019/06/12 职场文书
php引用传递
2021/04/01 PHP
MySQL 存储过程的优缺点分析
2021/05/20 MySQL
浅谈Redis跟MySQL的双写问题解决方案
2022/02/24 Redis