DoubleLi

qq: 517712484 wx: ldbgliet

  博客园 :: 首页 :: 博问 :: 闪存 :: 新随笔 :: 联系 :: 订阅 订阅 :: 管理 ::
  4737 随笔 :: 2 文章 :: 542 评论 :: 1615万 阅读
< 2025年3月 >
23 24 25 26 27 28 1
2 3 4 5 6 7 8
9 10 11 12 13 14 15
16 17 18 19 20 21 22
23 24 25 26 27 28 29
30 31 1 2 3 4 5

什么是属性?下面的例子a和b是属性吗?不是,他们是全局变量,属性(attribute)是类中的成员变量,也可以理解为属性就是类变量。

python
a = 11234
b = 'python'
PYTHON 复制 全屏

类中的变量是静态变量,类可以直接访问,python是一门动态语言,任何实例对象都可以动态地添加或删除属性,一个类定义了一个作用域,类实例对象也引入了一个作用域,这与类定义的作用域是不同的。在类实例对象中查找属性的时候,首先在实例自己的作用域中查找,如果没有找到,则再去类定义的作用域中查找。在对类实例属性进行赋值的时候,实际上会在类实例定义的作用域中添加一个属性或修改一个属性,但并不会影响到对应类中定义的同名属性。那么从访问属性到返回结果的过程是怎么运作的呢?是通过下面几个魔术方法来实现的。

相关方法的使用

  • __getattribute__:查找属性时会先触发该方法进行属性查找
  • __getattr__:查找属性没找到的时候触发
  • __setattr__:设置属性的时候触发
  • __delattr__:删除属性的时候触发

为了直观地感受实例访问属性时都做了什么,我们看一下下面的例子:

python
class TestCase:
    att_1 = 'hello'  # 定义类属性
    att_2 = 'python'

    def test_func(self):  # 定义方法
        print("这是一个方法")

    def __getattribute__(self, item):
        # 属性访问拦截器:当对象访问属性时,会自动触发这个方法,由这个方法来决定返回的属性值
        # 应用场景:访问不存在的属性时,不希望它报错,可以用try--except来捕获异常返回一个值或提示信息
        # try:
        #     return super().__getattribute__(item)  # 调用父类真正的__getattribute__方法返回正确的属性值
        # except AttributeError:
        #     print(item,':该属性不存在!')
        # 如果使用了try方法就不会触发__getattr__方法了,因为找不到属性时的异常已经在这里被捕获了
        print("我是__getattribute__,我正在工作")
        return super().__getattribute__(item)    # 调用父类的方法,返回找到的结果,不调用就不会返回

    def __getattr__(self, att_name):
        # 当对象访问属性时,属性不存在,引发异常,会被__getattr__方法捕获
        # 然后执行该方法的代码,相当于自带捕获异常
        print("我是__getattr__,我正在工作")
        return att_name + "这是我要找的东西,但是我找不到"

    def __setattr__(self, att_name, value):
        # 设置属性的时候就会触发该方法
        print("我是__setattr__,我正在工作")
        super().__setattr__(att_name, value)    # 调用父类的方法,设置属性,不调用就不会真的设置属性

    def __delattr__(self, att_name):
        print("我是__delattr__,我正在工作")
        print("这是我即将删除的东西{}".format(att_name))
        super().__delattr__(att_name)   # 调用父类的方法,删除属性,不调用就删除不了


res = TestCase()  # 实例化对象
print('-----------------访问属性----------------')
print(res.att_1)    # 访问类属性,通过运行结果看res实例对象访问属性时通过了方法__getattribute__

print('\n-----------------访问不存在的属性----------------')
print(res.att_3)

print('\n-----------------设置属性----------------')
res.att_3 = 'new_attr'

print('\n-----------------删除属性----------------')
del res.att_3

运行结果:

shell
C:\software\python\python.exe D:/learn/python/test.py
-----------------访问属性----------------
我是__getattribute__,我正在工作
hello

-----------------访问不存在的属性----------------
我是__getattribute__,我正在工作
我是__getattr__,我正在工作
att_3这是我要找的东西,但是我找不到

-----------------设置属性----------------
我是__setattr__,我正在工作

-----------------删除属性----------------
我是__delattr__,我正在工作
这是我即将删除的东西att_3

Process finished with exit code 0

​ 从结果中,我们可以看到当实例访问属性时,就会自动触发__getattribute__()方法,然后由这个方法来决定返回的属性值;当实例访问不存在的属性时,会引发异常,而这个异常会被__getattr__()方法捕获,然后执行该方法的代码,相当于自带捕获异常;同理,在设置属性或删除属性时,会自动触发对应的__setattr__()方法或__delattr__()方法。

​ 但需要注意的是,在自定义这些方法的时候,一定要记得调用对应的父类方法,将结果返回,否则只是执行了你自定义的东西并没有真正去做它本身要去做的事情。

动态属性设置

​ setattr()是python的一个内置函数,用于动态设置实例属性,语法:setattr(object, name, value)

​ 🍊 参数1:object-对象

​ 🍋 参数2:name-给对象要设置的属性名(字符串类型)

​ 🍍 参数3:value-属性值

python
class TestCase:
    """这是一个存放测试用例数据的类"""
    pass


cases = [
    {'case_id': 1, 'data': '123', 'actual': '不通过', 'excepted': '通过'},
    {'case_id': 2, 'data': '123', 'actual': '通过', 'excepted': '通过'},
    {'case_id': 3, 'data': '123', 'actual': '不通过', 'excepted': '通过'},
    {'case_id': 4, 'data': '123', 'actual': '通过', 'excepted': '通过'},
]

res = []  # 新建空列表用于存放结果
for i in cases:
    case = TestCase()  # 创建一个实例对象
    for k, v in i.items():
        # 把毎条用例数据中字典的key和value设成该实例的属性与值,比如:case_id = 1,data = 123等,把一个字典的全部键值对设为一个实例的属性
        setattr(case, k, v)  
    res.append(case)    # 把创建的对象存到结果中

print(res)  # 存放的是对象,共4个
print("\n按字段读取第一条测试用例数据:")
print("case_id:", res[0].case_id)   # 读取第一个对象的case_id属性值
print("data:", res[0].data)
print("actual:", res[0].actual)
print("excepted:", res[0].excepted)

print("\n读取完整的第一条测试用例数据:")
print(res[0].__dict__)

运行结果:

shell
C:\software\python\python.exe D:/learn/python/test.py
[<__main__.TestCase object at 0x0000019F15F573C8>, <__main__.TestCase object at 0x0000019F15F57C18>, <__main__.TestCase object at 0x0000019F16009C18>, <__main__.TestCase object at 0x0000019F16009C50>]

按字段读取第一条测试用例数据:
case_id: 1
data: 123
actual: 不通过
excepted: 通过

读取完整的第一条测试用例数据:
{'case_id': 1, 'data': '123', 'actual': '不通过', 'excepted': '通过'}

Process finished with exit code 0

​ 上面的运行结果中,像<main.TestCase object at 0x0000019F15F573C8>这就是一个TestCase()类的实例对象,这是它的内存地址,setattr()动态设置属性,就相当于case.case_id=1给实例设置一个属性,只是上面所举例是循环地把整个列表的数据拆分成4个实例的属性。

​ 同样的,python中对应还有一个getattr(object, name)内置函数,它是用于返回一个实例的属性,需要两个参数,一个是对象名,一个是属性名,返回该属性的属性值,这里不再举例,自己使用一下吧!

​ 另外,上述例子中最后还用到了一个__dict__对象属性,它是用于获取实例对象的所有属性,并以字典的形式返回。

posted on   DoubleLi  阅读(23)  评论(0编辑  收藏  举报
相关博文:
阅读排行:
· 阿里最新开源QwQ-32B,效果媲美deepseek-r1满血版,部署成本又又又降低了!
· 单线程的Redis速度为什么快?
· SQL Server 2025 AI相关能力初探
· AI编程工具终极对决:字节Trae VS Cursor,谁才是开发者新宠?
· 展开说说关于C#中ORM框架的用法!
历史上的今天:
2022-02-23 shell脚本中一些特殊符号
2022-02-23 Linux tr命令使用方法
2022-02-23 Linux Shell脚本中获取本机ip地址方法
2022-02-23 grep -v、-e、-E
2021-02-23 WebRTC直播课堂实践:实时互动是核心
2021-02-23 WebRTC媒体协商及实践
2021-02-23 爱奇艺直播 WebAssembly 优化之路
点击右上角即可分享
微信分享提示