Python 中@property的用法

在绑定属性时,如果我们直接把属性赋值给对象,比如:

p = Person()
p.name= 'Mary'

我们先看个详细的例子 (注意双下划线 name 和 age 定义为私有变量):

class Person(object):
def __init__(self, name, age):
self.__name = name
self.__age = age
def get_age_fun(self):
return self.__age
def set_age_fun(self, value):
if not isinstance(value, int):
raise ValueError('年龄必须是数字!')
if value < 0 or value > 100:
raise ValueError('年龄必须是 0-100')
self.__age = value
def print_info(self):
print('%s: %s' % (self.__name, self.__age))
p = Person('balala',20)
p.__age = 17
print(p.__age) # 17
print(p.get_age_fun()) # 20 表面上看,上面代码“成功”地设置了 __age 变量 17, 但实际上这个 __age 变量和 class 内部的 __age 变量不是一个变量!
# 内部的 __age 变量已经被 Python 解释器自动改成了 _Person_age,而外部代码给 p 新增了一个 __age 变量。 所以调用 get_age_fun 输出的是初始值
p.set_age_fun(35)
print(p.get_age_fun()) # 35
print(p.print_info()) # balala: 35

输出: 

17
20
35
balala: 35

表面上看,外部代码“成功”地设置了 __age 变量 17,但实际上这个 _age 变量和 class 内部的 _age 变量不是一个变量!
内部的 _age 变量已经被 Python 解释器自动改成了 _Person_age,而外部代码给 p 新增了一个 _age 变量。 所以调用 get_age_fun 输出的是初始值 20
而 set_age_fun 通过 class 内部改变了 age 变量值, 所以最终输出 balala: 35

我们再稍微调整下:

( 注意只改变了一个变量名: 原来的私有属性 __age 单下划线为: _age, 也可以定义为:age. 
解释: 以一个下划线开头的实例变量名,比如 _age,这样的实例变量外部是可以访问的,但是,按照约定俗成的规定,当看到这样的变量时,意思是,"虽然可以被访问,但是,请视为私有变量,不要随意访问。")
class Person(object):
    def __init__(self, name, age):
        self.__name = name
        self._age = age
</span><span style="color: rgba(0, 0, 255, 1)">def</span><span style="color: rgba(0, 0, 0, 1)"> get_age_fun(self):
     </span><span style="color: rgba(0, 0, 255, 1)">return</span><span style="color: rgba(0, 0, 0, 1)"> self._age

</span><span style="color: rgba(0, 0, 255, 1)">def</span><span style="color: rgba(0, 0, 0, 1)"> set_age_fun(self, value):
    </span><span style="color: rgba(0, 0, 255, 1)">if</span> <span style="color: rgba(0, 0, 255, 1)">not</span><span style="color: rgba(0, 0, 0, 1)"> isinstance(value, int):
        </span><span style="color: rgba(0, 0, 255, 1)">raise</span> ValueError(<span style="color: rgba(128, 0, 0, 1)">'</span><span style="color: rgba(128, 0, 0, 1)">年龄必须是数字!</span><span style="color: rgba(128, 0, 0, 1)">'</span><span style="color: rgba(0, 0, 0, 1)">)
    </span><span style="color: rgba(0, 0, 255, 1)">if</span> value &lt; 0 <span style="color: rgba(0, 0, 255, 1)">or</span> value &gt; 100<span style="color: rgba(0, 0, 0, 1)">:
        </span><span style="color: rgba(0, 0, 255, 1)">raise</span> ValueError(<span style="color: rgba(128, 0, 0, 1)">'</span><span style="color: rgba(128, 0, 0, 1)">年龄必须是0-100</span><span style="color: rgba(128, 0, 0, 1)">'</span><span style="color: rgba(0, 0, 0, 1)">)
    self._age </span>=<span style="color: rgba(0, 0, 0, 1)"> value

</span><span style="color: rgba(0, 0, 255, 1)">def</span><span style="color: rgba(0, 0, 0, 1)"> print_info(self):
    </span><span style="color: rgba(0, 0, 255, 1)">print</span>(<span style="color: rgba(128, 0, 0, 1)">'</span><span style="color: rgba(128, 0, 0, 1)">%s: %s</span><span style="color: rgba(128, 0, 0, 1)">'</span> % (self.<span style="color: rgba(128, 0, 128, 1)">__name</span><span style="color: rgba(0, 0, 0, 1)">, self._age))

p = Person('balala',20)
p._age
= 17
print(p._age) # 17
print(p.get_age_fun()) # 这里是 17 不再是 20, 因为此时 _age 是全局变量, 外部直接影响到类内部的更新值

p.set_age_fun(
35)
print(p.get_age_fun()) # 35

print(p.print_info()) # balala: 35

输出:

1 17
2 17
3 35
4 balala: 35

看的出私有和全局的设置

但是,上面的调用方法是不是略显复杂,没有直接用属性这么直接简单。

有没有可以用类似属性这样简单的方式来访问类的变量呢?必须的,对于类的方法
我们先来看一个稍微改造的例子:(稍后我们再使用 Python 内置的@property装饰器就是负责把一个方法变成属性调用.)

我们进入正题:看看 @property 的妙用之处:

 1 class Person(object):
 2     def __init__(self, name, age):
 3         self.__name = name
 4         self.__age = age
 5 
 6     @property
 7     def get_age_fun(self):
 8          return self.__age
 9 
10     @get_age_fun.setter # get_age_fun 是上面声明的方法
11     def set_age_fun(self, value):
12         if not isinstance(value, int):
13             raise ValueError('年龄必须是数字!')
14         if value < 0 or value > 100:
15             raise ValueError('年龄必须是 0-100')
16         self.__age = value
17 
18     def print_info(self):
19         print('%s: %s' % (self.__name, self.__age))
20 
21 
22 p = Person('balala',20)
23 p.__age = 17
24 print(p.__age) # 17
25 print(p.get_age_fun) # 20 注意这里不带 ()
26 
27 #p.set_age_fun(35) 注意不能这样调用赋值了
28 p.set_age_fun = 35 #  这里 set_age_fun 就是 声明的函数不带 ()
29 print(p.get_age_fun) # 35
30 print(p.print_info()) # balala: 35

输出:

17
20
35
balala: 35