在绑定属性时,如果我们直接把属性赋值给对象,比如:
p = Person()
p.name= 'Mary'
那么会发生什么事情呢?我们可以按照下面的步骤做个试验。
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
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输出的是初始值。
那我们稍微调整一下:
(注意只改变了一个变量名: 原来的私有属性 __age 单下划线为: _age,也可以定义为:age.
解释:以一个下划线开头的实例变量名,比如_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()) # 这里是17 不再是 20,因为此时_age是全局变量,外部直接影响到类内部的更新值
p.set_age_fun(35)
print(p.get_age_fun()) # 35
print(p.print_info()) # balala: 35
输出:
17
17
35
balala: 35
从上面两个例子中可以看出私有变量和全局变量的区别。
但是,上面的调用方法略显复杂,没有直接用属性这么直接简单。
有没有可以用类似属性这样简单的方式来访问类的变量呢?必须的,对于类的方法,我们来看一个改造的例子:(稍后我们再使用Python内置的@property装饰器就是负责把一个方法变成属性调用)
class Person(object):
def __init__(self, name, age):
self.__name = name
self.__age = age
@property
def get_age_fun(self):
return self.__age
@get_age_fun.setter # get_age_fun是上面声明的方法
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 注意这里不带()
#p.set_age_fun(35) 注意不能这样调用赋值了
p.set_age_fun = 35 # 这里set_age_fun 就是 声明的函数不带()
print(p.get_age_fun) # 35
print(p.print_info()) # balala: 35
输出:
17
20
35
balala: 35
大家自己理解这个例子吧~