Python之属性、特性和修饰符

1.property和@property还是直接上代码来的方便property()用法@property用法2.__getattribute__()、__getattr__()、__set

大家好,又见面了,我是全栈君,今天给大家准备了Idea注册码。

全栈程序员社区此处内容已经被作者隐藏,请输入验证码查看内容
验证码:
请关注本站微信公众号,回复“验证码”,获取验证码。在微信里搜索“全栈程序员社区”或者“www_javaforall_cn”或者微信扫描右侧二维码都可以关注本站微信公众号。

作为面对对象的核心内容,将从以下一个方面进行总结:

  1. property和@property

  2. __getattribute__()、__getattr__()、__setattr__()、__delattr__()

  3. 描述符__get__()、__set__()、__delete__()

  4. 不可变对象的实现

1. property和@property

  还是直接上代码来的方便

  property()用法

#coding = utf-8

class property_test:
    def __init__(self, size=10):
        self.size=size
        
    def __getsize__(self):
        return self.size

    def __setsize__(self, value):
        self.size = value

    def __deletesize__(self):
  
        del self.size

    x= property(__getsize__, __setsize__, __deletesize__)    

if __name__ == "__main__":
    p = property_test()
    p.x = 20
    print (p.size)  >>> 20

   @property用法

#coding = utf-8

'''class property_test:
    def __init__(self, size=10):
        self.size=size
        
    def __getsize__(self):
        return self.size

    def __setsize__(self, value):
        self.size = value

    def __deletesize__(self):
  
        del self.size

    x= property(__getsize__, __setsize__, __deletesize__)   '''

class property_test:
    def __init__(self, size=10):
        self.sizevalue=size     #特别注意sizevalue不要和size同名

    @property
    def size(self):
        return self.sizevalue

    @size.setter
    def size(self, value):
        self.sizevalue = value

    @size.deleter
    def size(self):
        del self.sizevalue

if __name__ == "__main__":
    p = property_test()  
    p.size = 20
    p.size += 20
    print (p.size, p.sizevalue)

2. __getattribute__()、__getattr__()、__setattr__()、__delattr__()

  属性查找过程:

  __getattribute__  >>>> __dict__  >>>> __slots__ >>>> __getattr__

  主要注意__getattr__的用法和无限递归错误

  __getattr__在实例以及对应的__dict__中查找属性失败,那么会调用__getattr__函数

  防止无限递归错误可以使用以下两种方法:

   1. self.__dict__[name] = value
        2. super().__setattr__(name,value)

  例:

class rectange:
    def __init__(self, width=20, height=40):
        self.width = width
        self.height = height

    def __getattr__(self, name):
        return self.width
        
    def __setattr__(self,name, value):
        if name == "squ":
            super().__setattr__(name,value)
            super().__setattr__("width",40)
            super().__setattr__("height",40)
            print (name, self.__dict__[name])
        else:
            #self.__dict__[name] = value
            super().__setattr__(name,value)
            print (name, self.__dict__[name])
    
if __name__ == "__main__":
    p = rectange()
    p.width = 10
    p.squ = 80
    print (p.width, p.height)

3. 描述符

  描述符说的直白点就是一个“绑定行为”的对象属性,通过__get__()、__set__()和__delete__()函数重写描述符属性

  如果上述方法被定义在一个对象中,那么这个对象就是一个描述符

  当定义拥有者类时,每个修饰符对象都是修饰符类的实例,绑定在类级别的属性上

  方法的原型为:

  ① __get__(self, instance, owner)

  ② __set__(self, instance, value)

  ③ __del__(self, instance)

class A:
    def __init(self):
        self.color = "red"
        
    def __get__(self, instance, owner):
        print ("get instance:", instance)
        print ("get owner:", owner)
        return self.color

    def __set__(self, instance, value):
   
        print ("set instance:", instance)
        print ("set value:", value)
        self.color = value

    def __delete__(self, instance):

        print ("delete instance:", instance)
        del self.color

class B:
    a = A()
    def __init__(self):
        self.color = "black"
    
if __name__ == "__main__":
    b = B()
    b.a = "blue"
    b.color = b.a
    del b.a

Python之属性、特性和修饰符

4. 使用__slots__实现不可变对象

  首先说明下__slots__方法,具体可以查看https://www.liaoxuefeng.com/wiki/001374738125095c955c1e6d8bb493182103fac9270762a000/0013868200605560b1bd3c660bf494282ede59fee17e781000

  方法:

  1. 把__slots__设为唯一允许操作的属性,这会使得对象内部的__dict__不再有效并阻止对其他属性的访问

  2. 在__init__中调用基类的__setattr__实现

  3. 在__setattr__中抛出异常

  例:

#coding=utf-8

class student:
    __slots__ = ("name")

    def __init__(self, name):
        super().__setattr__("name", name)
        self.name = name
        print (self.name)

    def __str__(self):
        return "{0.name}".format(self)

    def __setattr__(self, name, value):
        raise Exception(" '{__class__.__name__}' has no\
        attribute '{name}'".format(__class__=self.__class__,name=self.name))

if __name__ == "__main__":
    s = student("zhanglin")
    #s.name = "lizhi"

__slot__定义类中可以被外界访问的属性,类似node中的exports。

当父类中定义了__slot__时,不能向父类中添加属性。如果子类中没有定义__slot__,则子类不受父类__slot__定义的限制。

如果父类与子类中都定义了__slot__,则影响的结果为父类与子类__slot__的合集。

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/120190.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • python激活码2021【2021.8最新】

    (python激活码2021)好多小伙伴总是说激活码老是失效,太麻烦,关注/收藏全栈君太难教程,2021永久激活的方法等着你。https://javaforall.net/100143.htmlIntelliJ2021最新激活注册码,破解教程可免费永久激活,亲测有效,上面是详细链接哦~S32PGH0SQB-eyJsaWNlbnNlSWQiOi…

    2022年3月26日
    94
  • 如何设置vimrc 配置文件_居中怎么设置

    如何设置vimrc 配置文件_居中怎么设置#HowToVimrc在炮制你自己的.vimrc配置时,这里只有一个原则.不要把任何你不理解的一行放到你的vimrc中.互联网上有以吨计的入门手册,诸如这篇文章.它们之中包含各种令人惊叹的Hack技巧,这可以记你的Vim更好,但为了让你的环境更好完全从别人那儿照搬那些配置,完全是**最糟糕**的做法.真真切切地花一些时间来学习你

    2022年5月3日
    63
  • cmd进入mysql的方法

    cmd进入mysql的方法1.cmd下找到mysql的安装目录下的bin文件夹(可以直接在windows的bin文件夹下敲入cmd回车) 例如:E:\ProgramFiles(x86)\Wamp\bin\mysql\mysql5.5.202.运行mysql.exe或者mysql-hlocalhost-uroot-p3.输入root用户密码

    2022年6月8日
    35
  • C语言数组初始化

    C语言数组初始化转载博客代码编译运行环境:VS2017+Win32+Debug1.字符数组的初始化方式C语言中表示字符串有两种方式,数组和指针,字符数组是我们经常使用的方式。变量的定义包括指明变量所属类型、变量名称、分配空间以及初始化。可以看出,变量的初始化是变量定义的一部分。除了const变量需要显示初始化以外,其它变量如果在定义时未显示初始化,编译器会为变量以默认…

    2022年7月18日
    19
  • Error running’ xxxxxx’: Command line is too long. Shorten command line for xxxxxxxxx

    Error running’ xxxxxx’: Command line is too long. Shorten command line for xxxxxxxxx在IntellIDEA运行main函数的时候遇到了如下错误:Errorrunning’xxxxxx’:Commandlineistoolong.Shortencommandlineforxxxxxxxxx后来从网上查找资料发现是命令行的参数太长,需要进行修改。修改过程如下:在run->editconfiguration中修改红色画圈部分,选择如下…

    2022年8月22日
    18
  • 什么是线程死锁以及如何避免死锁「建议收藏」

    什么是线程死锁以及如何避免死锁「建议收藏」认识线程死锁多个线程同时被阻塞,他们中的一个或者全部都在等待某个资源被释放。由于线程被无限期地阻塞,因此程序不可能正常终止,最终导致死锁产生。如下图所示,线程A持有资源2,线程B持有资源1,他们同时都想申请对方的资源,所以这两个线程就会因互相等待而进入死锁状态。线程死锁示意图下面通过一个例子来说明线程死锁,代码模拟了上图的死…

    2022年7月15日
    16

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号