访问限制
.
- 描述
在一个模块中,会定义很多函数和变量,有的函数和变量允许别人使用,但有的函数和变量仅想在模块内使用,可以通过定义该函数、变量是公开的还是私有的来达到访问限制
.
- 私有变量(private)
class Student(object):
def __init__(self, name, score):
self.name = name
self.score = score
st = Student('maybe',25)
print(st.name)
# 输出结果:maybe
st.name = 'fy'
print(st.name)
# 输出结果:fy
这个Student类的定义来看,外部代码就可以自由地修改一个实例的name
属性
如果内部属性不想被外部访问,可以把属性的名称前加上两个下划线__
,在Python中实例的变量名如果以__开头,就变成了一个私有变量(private),只有内部可以访问,外部不能访问
class NewStudent(object):
def __init__(self, name, score):
self.__name = name
self.__score = score
def print_score(self):
print('{}: {}'.format(self.__name, self.__score))
st = NewStudent('maybe',25)
print(st.__name)
# 输出结果:AttributeError: 'NewStudent' object has no attribute '__name'
改动后已经无法从外部访问实例变量.__name
,通过访问限制的保护,使代码变得更加健壮
.
- 通过定义类方法访问内部属性
如果外部需要获取name属性,可以给NewStudent类增加get_name
方法
class NewStudent(object):
def __init__(self, name, score):
self.__name = name
self.__score = score
def get_name(self):
return self.__name
st = NewStudent('maybe',25)
print(st.get_name())
# 输出结果:maybe
如果外部需要修改score
,同样可以给NewStudent
类增加set_score
方法
class NewStudent(object):
def __init__(self, name, score):
self.__name = name
self.__score = score
def get_score(self):
return self.__score
def set_score(self,score):
self.__score = score
st = NewStudent('maybe',25)
print(st.get_score())
# 输出结果:25
st.set_score(30)
print(st.get_score())
# 输出结果:30
在类中定义set_score
方法目的是可以对参数做检查,避免传入无效的参数
class NewStudent(object):
def __init__(self, name, score):
self.__name = name
self.__score = score
def set_score(self,score):
if 0 <= score <= 100:
self.__score = score
else:
raise ValueError('bad score')
.
- 特殊变量
在Python中以双下划线开头,并且以双下划线结尾的,是特殊变量,变量名类似__xxx__
,特殊变量是可以直接访问的,不是私有变量
class NewStudent(object):
def __init__(self, name, score):
self.__name__ = name
self.__score__ = score
st = NewStudent('maybe',25)
print(st.__name__)
# 输出结果:maybe
.
- 单下划线情况
有些时候,可能遇到一个下划线开头的实例变量名,比如_name
,这样的变量外部是可以直接访问的,但是,按照俗称的规定,当你看到这样的变量时,意思就是’我虽然可以被访问,但是,请把我看做私有变量,不要随意访问‘
.
- 私有变量不能访问的原理
双下划线开头的实例变量并不是完全访问不了。不能直接访问的原因是__name
是因为Python解释器对外把__name
变量改成了_NewStudent__name
,如果我们访问_NewStudent__name
来访问__name
变量还是可以实现的
class NewStudent(object):
def __init__(self, name, score):
self.__name = name
self.__score = score
def print_score(self):
print('{}: {}'.format(self.__name, self.__score))
st = NewStudent('maybe',25)
print(st._NewStudent__name)
# 输出结果:maybe
Python本身没有任何机制阻止你干坏事,一切全靠自觉