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

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 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) #
print(p.get_age_fun()) # 这里是17 不再是 20,因为此时_age是全局变量,外部直接影响到类内部的更新值 p.set_age_fun(35)
print(p.get_age_fun()) # print(p.print_info()) # balala: 35

输出:


 balala: 

看的出私有和全局的设置

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

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

我们进入正题:看看@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) #
print(p.get_age_fun) # 20 注意这里不带() #p.set_age_fun(35) 注意不能这样调用赋值了
p.set_age_fun = 35 # 这里set_age_fun 就是 声明的函数不带()
print(p.get_age_fun) #
print(p.print_info()) # balala: 35

输出:


balala: 

最新文章

  1. CentOS中Mysql常用操作
  2. C#获得MAC地址(网卡序列号)代码
  3. jQuery - 全国省市县三级联动
  4. jsp2.0+中的标签文件,JSP Fragment技术
  5. Docker中搭建Hadoop-2.6单机伪分布式集群
  6. MVC:上传文件时限制文件类型
  7. java 部署服务报:Bad version number in .class file
  8. 让内层Div将外层Div撑开
  9. android ptl
  10. firefly的环境搭建(2013年9月25日最新,win下最详图文)
  11. [AS/400] Control Language
  12. load_library(linker.cpp:759): library &quot;libmaliinstr.so&quot; not found
  13. iOS 开发中中 textView 作为子控件点击输入文本,然后退出文本的方式
  14. springboot部分常用注解
  15. java中的闭包
  16. servlet的xx方式传值中文乱码
  17. F#周报2018年第49期
  18. HDU 4508 湫湫系列故事——减肥记I (完全背包)
  19. HDU 3455
  20. REST 服务器调试 RESTDebugger.exe 和浏览器测试

热门文章

  1. maven包加载
  2. C语言 &gt; 数组和指针
  3. Egg Dropping Puzzle问题的分析
  4. https://doc.opensuse.org/projects/kiwi/doc/
  5. Hibernate中的多表查询及抓取策略
  6. 使用 Java8 Optional 的正确姿势(转)
  7. OAthe2 Login use OkHttpClient and OAuth2RestTemplate
  8. Spring提供的用于访问Rest服务的客户端:RestTemplate实践
  9. 苹果通知推送服务(APNS)一些关键特性摘要
  10. 利用java反射机制实现读取excel表格中的数据