python属性property

属性函数(property)
在对象中两个很重要的元素就是属性和方法,在调用的时候两者是有区别的。

 class People:
     def __init__(self,first_name,last_name):
         self.first_name = first_name
         self.last_name  = last_name

     def get_first_name(self):
         return self.first_name

a = People('lala','ouyang')
print(a.get_first_name())
print(a.first_name)

从例子中我们可以发现,一样的结果,但是调用的过程不一样(虽然其实也就是多一个括号而已),那么有没有一种办法,使得我们调用属性的时候就会自动调用相应的方法,也就是增加一些额外的处理过程(例如类型检查或者验证)。这时候属性函数(@property)就能给我们提供很好的解决方案。

首先是最简单的例子,自动调用get,set函数对属性的处理。

class People:
    def __init__(self,name):
        self.name = name

    #getter function
    @property #属性函数
    def name(self):
        return self._name

    #setter function
    @name.setter
    def name(self,name):
        self._name = name


a = People('leida')
print(a.name) 
a.name = 'libai' 
print(a.name)

正如例子中这样。要定义对属性的访问,一种最简单的方法就是将其定义为property。比如说,增加对属性的类型检查:

class People:
    def __init__(self,name):
        self.name = name

    #getter function
    @property #属性函数
    def name(self):
        return self._name

    #setter function
    @name.setter
    def name(self,name):
        if not isinstance(name,str):
            raise TypeError('name must is string type')
        self._name = name

a = People(12)

当我们实现一个property时,底层数据仍然需要保存在某个地方,因而在get和set的方法中,可以看到直接对_name操作的,这就是数据实际保存的地方。但是,也发现在init()方法中任然是对self.name操作的。但是实际情况是我初始化的时候程序仍旧是对self._name操作的。(这点我也还不理解,应该不是这样的啊.但是必须这么写,不然会报错)。
对于已经存在的get,set方法,同样也可以定义为property:

class People:
    def __init__(self,name):
        self.name = name

    def get_name(self):
        print('calling the get function')
        return self._name

    def set_name(self,name):
        print('calling the set function')
        self._name = name

    name = property(get_name,set_name,del_name)

a = People('libai')

同时,在set_name 函数中做了打印标记,发现在init()方法中确实调用了set_name()函数。

Property属性实际上就是把一系列的方法绑定到一起。如果检查类的property属性,就会发现property自身所持有的属性fget,fset所代表的原始方法。

print(People.name.fget)
print(People.name.fset)

一般来说,我们不会直接去调用fset或者fget,但是当我们调用property属性时会自动触发对这些方法的调用。
上面例子中的两种写法,一般倾向于第二种写法,特别是如果需要对某个普通的属性额外增加处理步骤时,可以在不修改已有代码的情况下将这个属性提升为一个property。
Property也可以用来定义需要计算的属性,这类属性并不会实际被保存起来,而是根据需要完成计算。

import math
class Circle:
    def __init__(self,radius):
        self.radius = radius

    @property
    def area(self):
        return math.pi * self.radius**2
    @property
    def perimeter(self):
        return 2*math.pi*self.radius

c = Circle(5)
print(c.area)
print(c.perimeter)

这样的写法就会使得实例的接口变得统一,本来用方法实现的计算调用的时候用属性就可以,很好的避免了方法、属性傻傻分不清的情况了。
个人建议,不要在代码中不断重复使用@property,这样会使得代码变得臃肿,而且难以阅读,容易出错。同样的任务,利用描述符或者闭包也能够很好的完成。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 135,269评论 19 139
  • 函数和对象 1、函数 1.1 函数概述 函数对于任何一门语言来说都是核心的概念。通过函数可以封装任意多条语句,而且...
    道无虚阅读 4,692评论 0 5
  • 转至元数据结尾创建: 董潇伟,最新修改于: 十二月 23, 2016 转至元数据起始第一章:isa和Class一....
    40c0490e5268阅读 1,857评论 0 9
  • 人们好多都说那个人怎么怎么怎么,她又得奖了,他拿了奖学金,他的这个月的工资好高,经理又表扬他了,等等!可是这关你什...
    耳语心声阅读 1,011评论 0 1
  • 当今的人们压力太大,对年轻人来说:独身的时候忙于找对象,焦虑不安,没法放松;成家以后,要买房买车、生儿...
    晚起画蛾眉阅读 1,771评论 3 2