Python的特性(Property)和描述符

特性(Property)

特性是对类的一个特定属性进行拦截,在操作这个属性时,执行特定的函数,对属性的操作进行拦截。

特性的实现

特性使用property类来实现,也可以使用property装饰器实现,二者本质是一样的。

property类的__init__函数接收4个参数,来实现属性的获取、赋值、删除及文档。

    def __init__(self, fget=None, fset=None, fdel=None, doc=None): # known special case of property.__init__
        """
        property(fget=None, fset=None, fdel=None, doc=None) -> property attribute

        fget is a function to be used for getting an attribute value, and likewise
        fset is a function for setting, and fdel a function for del‘ing, an
        attribute.  Typical use is to define a managed attribute x:

        class C(object):
            def getx(self): return self._x
            def setx(self, value): self._x = value
            def delx(self): del self._x
            x = property(getx, setx, delx, "I‘m the ‘x‘ property.")

        Decorators make defining new properties or modifying existing ones easy:

        class C(object):
            @property
            def x(self):
                "I am the ‘x‘ property."
                return self._x
            @x.setter
            def x(self, value):
                self._x = value
            @x.deleter
            def x(self):
                del self._x

        # (copied from class doc)
        """
        pass

从代码上看,4个参数都不是必须的,如果没有传入对应的操作函数,则取默认值None,则对应的操作不受支持,试图调用默认值None时,会引发异常。

测试代码:

class Man(object):

    def __init__(self, age):
        self._age = age

    @property
    def age(self):
        return self._age

    @age.getter
    def age(self):
        return self._age

if __name__ == ‘__main__‘:

    tom = Man(22)
    print(tom.age)
    tom.age = 23
    print(tom.age)

因为缺少setter的函数方法,所以试图给age赋值时,会引发异常。

AttributeError: can‘t set attribute

age()方法上增加@property装饰器,实际上等同于age = property(),将age赋值为property的实例。

经过测试,把

@property
def age(self):
    pass

修改为

age = property()

不影响代码的执行,所以@age.setter装饰器就很好理解了,因为age是property类的实例,property类有setter的方法,age继承了property类的setter方法,所以可以使用@age.setter装饰器。其他的setter、getter、deleter也是同理。

特性的继承

类的实例和子类都会继承类的特性,测试代码:

class Person(object):  def __init__(self, age):
        self._age = age

    @property
    def age(self):
        return self._age

    @age.setter
    def age(self, value):
        self._age = value * 2

    @age.getter
    def age(self):
        return self._age

class Man(Person):
    pass

if __name__ == ‘__main__‘:

    tom = Man(22)
    print(tom.age)
    tom.age = 23
    print(tom.age)

在age的setter方法中,将age的值乘以2,从运行结果上看,子类及其实例都是继承property的

22
46

描述符

任何带有如下方法的类都可以作为描述符

def __get__(self, instance, owner):
    ...

def __set__(self, instance, value):
    ...

def __delete__(self, instance):
    ....

__get__方法接收三个参数:self为描述符实例自身; instance指访问属性所属的实例,当访问的属性为类属性时,instance为None;owner指描述符实例附加到的类。

测试代码:

class Descriptor:

    def __init__(self):
        pass

    def __get__(self, instance, owner):
        print(self, instance, owner)

    def __set__(self, instance, value):
        print(self, instance, value)

    def __delete__(self, instance):
        print(self, instance)

class ClassA(object):

    a = Descriptor()

if __name__ == ‘__main__‘:
    ClassA.a
    x = ClassA()
    x.a

从运行结果可以看出,当访问类属性时,instance为None;当访问实例属性时,instance为属性所属的的实例。

instance属性为None时,描述符对象会直接访问所附加到的类的类属性。当instance属性不为None时,描述符会直接访问实例属性。

<__main__.Descriptor object at 0x1081413c8> None <class ‘__main__.ClassA‘>
<__main__.Descriptor object at 0x1081413c8> <__main__.ClassA object at 0x108141358> <class ‘__main__.ClassA‘>

__set__方法的第三个参数value为需要赋值的属性,__delete__方法的两个参数与__get__方法的前两个参数相同。

时间: 2024-08-27 04:20:31

Python的特性(Property)和描述符的相关文章

python中名称修饰与描述符

名称修饰 java和C#等其他高级语言中都有private关键字来修饰一个属性或字段是私有的,但是python中并没有private,而是有个与它接近的概念旧式名称修饰.每当在一个属性前面加上__前缀,解释器就会立刻将其重命名: 直接访问会抛异常 利用dir函数查看内部属性 python内部会把__前缀的属性重命名为[_类名+属性名]:因此在python中如果一个属性不是共有的就约定使用双下划线__为前缀,它不会调用任何名称修饰的算法,只是说名这个属性是该类的私有属性. 幸运的是python中还

python进程间传递文件描述符扩展库

由于python本身的线程基本上比较残废,所以为了利用机器的cpu,就不得不用上多进程... 在游戏服务器的设计中,最为常见的方式是: 挂一个前端服务器,专门来维护与客户端的连接,然后将客户端的请求数据转发给后端服务器... 上面的方式是现在最为正统的... 但是自己因为环境的限制,需要做到对客户端透明,然后将后端的服务器转换成为多进程的...所以这里就只有用一点比较别扭的方法了,首先处理登录等一些常规的逻辑放在前端服务器,当进入放进进行匹配战斗之后,将客户端的socket连接直接交给后端服务器

JS面向对象篇一、理解对象及属性特性(属性描述符)

本文内容 1.理解对象; 2.ECMAScript有两种属性类型:数据属性和访问器属性(getter和setter函数); 3.数据属性的属性特性:[[Configurable]].[[Enumerable]].[[Writable]].[[value]]; 4.访问器属性的属性特性:[[Configurable]].[[Enumerable]].[[get]].[[set]]; 5.Object.defineProperty().Object.defineProperties().Object

Python中的动态属性与描述符

动态属性与属性描述符 属性描述符是什么? ??在解释属性查找顺序之前我们需要了解Python中的属性描述符,属性描述符作为其他类对象的属性而存在,实现了特殊方法中的get.set.delete中的一种即可称作属性描述符. 其中只实现了__get__()的称作非数据描述符,实现了__get__()和__set__()方法的称作数据描述符. Data.py class Data(): def __get__(self, instance, owner): pass def __set__(self,

Python——@property属性描述符

@property 可以将python定义的函数“当做”属性访问,从而提供更加友好访问方式,但是有时候setter/getter也是需要的 假设定义了一个类Cls,该类必须继承自object类,有一私有变量__x 1. 第一种使用属性的方法: ? 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 #!/usr/bin/env python # -*- coding: utf-8 -*- # blog.i

ECMAScript5新特性之属性描述符读写对象的属性

直接读写.使用Getter/Setter accessors 'use strict';var fruit = {}; Object.defineProperties(fruit,{ 'name' : { configurable : true, enumerable : true, get : function(){ console.log('Getter方法'); return name; }, set : function(val){ console.log('Setter方法'); na

python 描述符 上下文管理协议 类装饰器 property metaclass

1.描述符 #!/usr/bin/python env # coding=utf-8 # 数据描述符__get__ __set__ __delete__ ''' 描述符总结 描述符是可以实现大部分python类特性中的底层魔法,包括@classmethod,@staticmethd,@property甚至是__slots__属性 描述符是很多高级库和框架的重要工具之一,描述符通常是使用到装饰器或者元类的大型框架中的一个组件 注意事项: 一 描述符本身应该定义成新式类,被代理的类也应该是新式类 二

python特性property

通常,访问类和实例属性的时候,将返回所存储的相关值,也就是直接和类(实例的)的__dict__打交道.若果要规范这些访问和设值方式的话, 一种方法是数据描述符,另一种就是python内置的数据描述符协议函数Property().property是一种特殊的值,访问它时会计算它的值. 特性的原型函数是property(getf=None,setf=None,delf=None,doc=None),函数的前三个参数分别对应描述符的__get__.__set__.__delete__方法. class

【Python】【元编程】【二】【描述符】

""" #描述符实例是托管类的类属性:此外,托管类还有自己实例的同名属性 #20.1.1 LineItem类第三版:一个简单的描述符#栗子20-1 dulkfood_v3.py 使用 Quantity 描述符管理 LineItem 的属性class Quantity:# 描述符基于协议实现,无需创建子类. def __init__(self,storage_name): self.storage_name = storage_name def __set__(self, in