17、类的自定义属性访问及动态属性设置

软件发布|下载排行|最新软件

当前位置:首页IT学院IT技术

17、类的自定义属性访问及动态属性设置

miki_peng   2020-02-23 我要评论

前言:本文主要介绍python类的一些自定义属性访问的方法,以及类的动态属性设置即python的内置函数setattr()

自定义属性访问

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

a = 11234
b = 'python'

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

相关方法的使用

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

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

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

运行结果:

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)

Copyright 2022 版权所有 软件发布 访问手机版

声明:所有软件和文章来自软件开发商或者作者 如有异议 请与本站联系 联系我们