Python进阶之“属性(property)”详解

Posted churao

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Python进阶之“属性(property)”详解相关的知识,希望对你有一定的参考价值。

Python中有一个被称为属性函数(property)的小概念,它可以做一些有用的事情。在这篇文章中,我们将看到如何能做以下几点:

  • 将类方法转换为只读属性
  • 重新实现一个属性的setter和getter方法

在本文中,您将学习如何以几种不同的方式来使用内置的属性函数。希望读到文章的末尾时,你能看到它是多么有用。

开始

使用属性函数的最简单的方法之一是将它作为一个方法的装饰器来使用。这可以让你将一个类方法转变成一个类属性。当我需要做某些值的合并时,我发现这很有用。其他想要获取它作为方法使用的人,发现在写转换函数时它很有用。让我们来看一个简单的例子:

class Person(object):
    """"""

    def __init__(self, first_name, last_name):
        """Constructor"""
        self.first_name = first_name
        self.last_name = last_name
    
    @property
    def full_name(self):
        """
        Return the full name
        """
        return "%s %s" % (self.first_name, self.last_name)
xxxxxxxxxx
14
 
1
class Person(object):
2
    """"""
3
4
    def __init__(self, first_name, last_name):
5
        """Constructor"""
6
        self.first_name = first_name
7
        self.last_name = last_name
8
    
9
    @property
10
    def full_name(self):
11
        """
12
        Return the full name
13
        """
14
        return "%s %s" % (self.first_name, self.last_name)
>>> person = Person("Mike", "Driscoll")
>>> person.full_name
‘Mike Driscoll‘
>>> person.first_name
‘Mike‘
>>> person.full_name = "Jackalope"
Traceback (most recent call last):
  File "<string>", line 1, in <fragment>
AttributeError: can‘t set attribute
xxxxxxxxxx
9
9
 
1
>>> person = Person("Mike", "Driscoll")
2
>>> person.full_name
3
‘Mike Driscoll‘
4
>>> person.first_name
5
‘Mike‘
6
>>> person.full_name = "Jackalope"
7
Traceback (most recent call last):
8
  File "<string>", line 1, in <fragment>
9
AttributeError: can‘t set attribute
>>> person.first_name = "Dan"
>>> person.full_name
‘Dan Driscoll‘
xxxxxxxxxx
1
>>> person.first_name = "Dan"
2
>>> person.full_name
3
‘Dan Driscoll‘

这是一种限制,因此让我们来看看另一个例子,其中我们可以创建一个允许设置的属性。

使用Python property取代setter和getter方法

让我们假设我们有一些遗留代码,它们是由一些对Python理解得不够好的人写的。如果你像我一样,你之前已经看到过这类的代码:

from decimal import Decimal
 
########################################################################
class Fees(object):
    """"""
    
    #----------------------------------------------------------------------
    def __init__(self):
        """Constructor"""
        self._fee = None
        
    #----------------------------------------------------------------------
    def get_fee(self):
        """
        Return the current fee
        """
        return self._fee
    
    #----------------------------------------------------------------------
    def set_fee(self, value):
        """
        Set the fee
        """
        if isinstance(value, str):
            self._fee = Decimal(value)
        elif isinstance(value, Decimal):
            self._fee = value
xxxxxxxxxx
27
 
1
from decimal import Decimal
2
 
3
########################################################################
4
class Fees(object):
5
    """"""
6
    
7
    #----------------------------------------------------------------------
8
    def __init__(self):
9
        """Constructor"""
10
        self._fee = None
11
        
12
    #----------------------------------------------------------------------
13
    def get_fee(self):
14
        """
15
        Return the current fee
16
        """
17
        return self._fee
18
    
19
    #----------------------------------------------------------------------
20
    def set_fee(self, value):
21
        """
22
        Set the fee
23
        """
24
        if isinstance(value, str):
25
            self._fee = Decimal(value)
26
        elif isinstance(value, Decimal):
27
            self._fee = value

要使用这个类,我们必须要使用定义的getter和setter方法:

>>> f = Fees()
>>> f.set_fee("1")
>>> f.get_fee()
Decimal(‘1‘)
xxxxxxxxxx
1
>>> f = Fees()
2
>>> f.set_fee("1")
3
>>> f.get_fee()
4
Decimal(‘1‘)

如果你想添加可以使用正常点符号访问的属性,而不破坏所有依赖于这段代码的应用程序,你可以通过添加一个属性函数非常简单地改变它:

from decimal import Decimal
 
########################################################################
class Fees(object):
    """"""
    
    #----------------------------------------------------------------------
    def __init__(self):
        """Constructor"""
        self._fee = None
        
    #----------------------------------------------------------------------
    def get_fee(self):
        """
        Return the current fee
        """
        return self._fee
    
    #----------------------------------------------------------------------
    def set_fee(self, value):
        """
        Set the fee
        """
        if isinstance(value, str):
            self._fee = Decimal(value)
        elif isinstance(value, Decimal):
            self._fee = value
            
    fee = property(get_fee, set_fee)
xxxxxxxxxx
29
1
from decimal import Decimal
2
 
3
########################################################################
4
class Fees(object):
5
    """"""
6
    
7
    #----------------------------------------------------------------------
8
    def __init__(self):
9
        """Constructor"""
10
        self._fee = None
11
        
12
    #----------------------------------------------------------------------
13
    def get_fee(self):
14
        """
15
        Return the current fee
16
        """
17
        return self._fee
18
    
19
    #----------------------------------------------------------------------
20
    def set_fee(self, value):
21
        """
22
        Set the fee
23
        """
24
        if isinstance(value, str):
25
            self._fee = Decimal(value)
26
        elif isinstance(value, Decimal):
27
            self._fee = value
28
            
29
    fee = property(get_fee, set_fee)

我们在这段代码的末尾添加了一行。现在我们可以这样做:

>>> f = Fees()
>>> f.set_fee("1")
>>> f.fee
Decimal(‘1‘)
>>> f.fee = "2"
>>> f.get_fee()
Decimal(‘2‘)
xxxxxxxxxx
 
1
>>> f = Fees()
2
>>> f.set_fee("1")
3
>>> f.fee
4
Decimal(‘1‘)
5
>>> f.fee = "2"
6
>>> f.get_fee()
7
Decimal(‘2‘)

正如你所看到的,当我们以这种方式使用属性函数时,它允许fee属性设置并获取值本身而不破坏原有代码。让我们使用属性装饰器来重写这段代码,看看我们是否能得到一个允许设置的属性值。

from decimal import Decimal
 
########################################################################
class Fees(object):
    """"""
    
    #----------------------------------------------------------------------
    def __init__(self):
        """Constructor"""
        self._fee = None
        
    #----------------------------------------------------------------------
    @property
    def fee(self):
        """
        The fee property - the getter
        """
        return self._fee
    
    #----------------------------------------------------------------------
    @fee.setter
    def fee(self, value):
        """
        The setter of the fee property
        """
        if isinstance(value, str):
            self._fee = Decimal(value)
        elif isinstance(value, Decimal):
            self._fee = value
            
#----------------------------------------------------------------------
if __name__ == "__main__":
    f = Fees()
xxxxxxxxxx
33
 
1
from decimal import Decimal
2
 
3
########################################################################
4
class Fees(object):
5
    """"""
6
    
7
    #----------------------------------------------------------------------
8
    def __init__(self):
9
        """Constructor"""
10
        self._fee = None
11
        
12
    #----------------------------------------------------------------------
13
    @property
14
    def fee(self):
15
        """
16
        The fee property - the getter
17
        """
18
        return self._fee
19
    
20
    #----------------------------------------------------------------------
21
    @fee.setter
22
    def fee(self, value):
23
        """
24
        The setter of the fee property
25
        """
26
        if isinstance(value, str):
27
            self._fee = Decimal(value)
28
        elif isinstance(value, Decimal):
29
            self._fee = value
30
            
31
#----------------------------------------------------------------------
32
if __name__ == "__main__":
33
    f = Fees()

上面的代码演示了如何为fee属性创建一个setter方法。你可以用一个名为@fee.setter的装饰器装饰第二个方法名也为fee的方法来实现这个。当你如下所做时,setter被调用:

>>> f = Fees()
>>> f.fee = "1"
xxxxxxxxxx
 
1
>>> f = Fees()
2
>>> f.fee = "1"

如果你看属性函数的说明,它有fget, fset, fdel和doc几个参数。如果你想对属性使用del命令,你可以使用@fee.deleter创建另一个装饰器来装饰相同名字的函数从而实现删除的同样效果。

结束语

现在你知道如何在你的类中使用Python的属性函数。希望你能找到更有用的方式,在你的代码中使用它们。




以上是关于Python进阶之“属性(property)”详解的主要内容,如果未能解决你的问题,请参考以下文章

Python高级property属性详解

python @property使用详解

Python之路:面向对象(进阶)

python语法基础-面向对象-进阶-长期维护

python0.16------构造函数/析构函数/self详解/重写/访问限制/对象属性和类属性/@property/运算符重载

Python: 浅淡Python中的属性(property)