封装
- 封装是指隐藏对象中一些不希望被外界所访问的属性和方法
- 隐藏属性:将对象的属性名修改为外部不知道的名字
- 使用封装增加了类定义的复杂程度,但确保了数据的安全性
本节主要内容有:
- 几种常见的隐藏属性的方法
- get( ) , set( ) 方法,
- @property装饰器 ,@属性名.setter 装饰器
# 封装是面向对象的三大特征之一
# 封装是指隐藏对象中一些不希望被外界所访问的属性和方法
# 隐藏属性:将对象的属性名修改为外部不知道的自己特有的名字
# getter 和 setter 方法:
# - getter 方法获取对象的指定属性(get_属性名)
# - setter 方法用来设置对象的指定属性(set_属性名)
# 使用封装增加了类定义的复杂程度,但确保了数据的安全性
# 1. 隐藏了属性名,使调用者无法随意修改对象中的属性
# 2. 通过getter,setter方法控制它是否可读写
# 3. 通过setter方法设置属性,可以增加数据的验证,确保数据(大小,类型)是正确的
# 4. 可以在使用getter,setter方法的同时做一些其他的处理
# 5. 使用getter方法可以表示一些计算的属性
class Dog :
def __init__(self, name,age ) :
self.dog_name = name
self.dog_age = age
def getter_name (self) :
return self.dog_name
def setter_name(self,name) :
self.dog_name = name
def getter_age(self) :
return self.dog_age
def setter_age(self,age) :
if age >= 0 :
self.dog_age = age
d = Dog('小黑',4)
d.setter_age(-3)
print( d.getter_name(),d.getter_age(),'岁')
# 如果想隐藏对象的属性,可以使用双下划线开头,__XXX
# 隐藏属性只能在类的内部访问,外部无法通过对象访问
# 实际上python内部是将 __xxx 改成了_类名__xxx 并不是绝对的隐藏
# 通常情况下,私有属性以 _xxx 的方式存储,没有限制修改,靠自觉
# 一般情况下,_开头的都是私有属性,没有特殊需求,不要修改私有属性
class Rectangle :
'''
表示矩形的类
'''
def __init__(self,width,height,a) :
self.hidden_width = width
self.hidden_height = height
self.__a = a # 外部无法直接访问 __a
def getter_a(self) :
return self.__a
def getter_width(self) :
return hidden_width
def setter_width(self,width) :
self.hidden_width = width
def getter_height(self) :
return hidden_height
def setter_height(self,height) :
self.hidden_height = height
def getter_area(self) :
return self.hidden_height * self.hidden_width
r = Rectangle(2,4,'双下划线')
r.setter_width(5)
r.setter_height(6)
print( r.getter_area() )
# print(r.__a) 直接访问会报错:'Rectangle' object has no attribute '__a'
print(r._Rectangle__a) # _类名__xxx 可以访问
r._Rectangle__a = '__能修改' # 也可以修改
print( r.getter_a() )
property装饰器,属性名.setter 装饰器:
class Person :
def __init__ (self,name,age) :
self._name = name
self._age = age
# property装饰器,用来将一个getter方法装换为对象的属性
# 添加property后,我们就可以像调用属性一样调用get方法
@property
def name(self):
return self._name
# 属性名.setter装饰器的前提是有同名的property装饰器
# 只有 property装饰器,没有 属性名.setter 时,直接使用 p.name='小刘' 会报错
@name.setter
def name (self,name ) :
self._name = name
@property
def age(self):
return self._age
@age.setter
def age (self,age ) :
self._age = age
p = Person ( '小王',18 )
print(p.name , p.age)
p.name = '小刘'
p.age = 20
print(p.name , p.age)