继续浏览精彩内容
慕课网APP
程序员的梦工厂
打开
继续
感谢您的支持,我会继续努力的
赞赏金额会直接到老师账户
将二维码发送给自己后长按识别
微信支付
支付宝支付

【Python入门】14.面向对象编程之 访问限制与绑定限制__slots__

慕神8447489
关注TA
已关注
手记 1310
粉丝 174
获赞 957

目录

面向对象编程
访问限制
绑定限制 __slots__
• 绑定属性与方法
• __slots__

面向对象编程

访问限制

我们把数据封装到class内部,从而把数据和执行逻辑隐蔽起来。但从class来看,我们依然能够从外部直接访问或修改内部的属性数据。

要让内部的属性数据不被外部访问,可以在属性名字前加双下划线__。在Python中,实例的变量名如果以__开头,就变成了一个私有变量(private),只有内部可以访问,外部不能访问。

class A(object):
    
    def __init__(self, name):        self.name = name                        #设置公开属性nameclass B(object):
    
    def __init__(self, name):                   #设置私有属性name
        self.__name = name         

a = A('a')
b = B('b')

从外部访问a和b的name属性:

>>>a.name
a>>>b.__nameAttributeError: 'B' object has no attribute '__name'             #显示b不存在__name属性

如果又想要外部代码允许访问和修改内部信息,那么可以添加get_name和set_name这样的方法

class B(object):
    
    def __init__(self, name):        self.__name = name    
    def get_name(self):                                 #添加一个返回实例__name属性的方法
        return self.__name    def set_name(self, name):                           #添加一个修改实例__name属性的方法
        self.__name = name

b = B('b')
>>>b.get_name()
b>>>b.set_name('abc')>>>b.get_name()
abc

用这种方法还有一个好处就是可以限制用户传入无效的数据

class B(object):
    
    def set_score(self,score):
        if 0 <= score <= 10:
            self.__score = score        else:            raise ValueError('bad score')

需要注意的是,在Python中,__xxx__这样的变量是特殊变量,可以从外部访问,不是private变量。

有时候会有以一个下划线开头命名的变量_xxx,这种变量可以从外部访问,但按约定俗成的习惯而言,这种变量被当成private变量,最好不要访问。

事实上,__xxx这种命名格式的变量也是可以访问的。不能直接访问的原因是python解释器把在class内部定义得__xxx变量自动改为_class名__name,我们依然可以通过后者来访问。

>>>b._B__name
abc

当然最好不要这样做。而且不同版本的python解释器会改成不一样的格式。

还要注意的是,不要在外部赋值给__name,看似可以成功,实际上,赋值的属性不是class内部定义的属性。如:

>>>b.__name = 'wow'>>>b.__name
wow>>>b.get_name()
abc

绑定限制 __slots__

• 绑定属性与方法

在定义一个class,创建对应的实例之后,我们就可以给实例绑定属性。

class Stu(object):
     passs = Stu()
s.name = 'Ming'

此外,还能给实例绑定方法,需要添加一个MethodType模块

def set_score(self,score):                               #先定义一个函数作为方法
    self.score = scorefrom types import MethodType                             #加入MethodType模块s.set_score = MethodType(set_score, s)                   #给实例a绑定set_score方法

调用这个绑定的方法

>>>s.set_score(90)                                       #调用上面绑定的方法>>>s.score                                               #访问a的score属性90

但是该绑定的方法只对实例a有效,对Stu类的其他实例无效。

>>>t = Stu()>>>t.set_score(99)AttributeError: 'Stu' object has no attribute 'set_score'

若想对Stu类的所有实例对象有效,给Stu绑定方法就可以了。

>>>Stu.set_score = set_score                             #把set_score方法绑定到Stu>>>s.set_score(90)>>>s.score  
90>>>>t.set_score(99)>>>t.score99
•  __slots__

如果要限制实例的属性,只允许添加规定的属性,那么可以在class内部定义一个__slots__变量,__slots__的值就是只允许添加的属性。

class Stu(object):
    __slots__ = ('name', 'score')                       #用tuple定义允许绑定的属性名称

效果如下:

>>> s = Stu()                                           #创建新的实例>>> s.name = 'Ming'                                     #成功绑定name属性>>> s.score = 90                                        #成功绑定score属性>>> s.gender = 'male'                                   #绑定gender属性失败AttributeError: 'Stu' object has no attribute 'gender'

需要注意的是,__slots__定义的属性仅对当前类实例有效,对继承的子类是无效的,如:

>>>class A(Stu):...    pass>>>t = A()>>>t.gender = 'female'                                  #成功绑定gender属性>>>t.gender
female

如果想要对子类也有效,那么子类的定义也要添加__slots__变量,这是子类的允许添加的属性包括子类的__slots__和父类的__slots__

>>>class A(Stu):...    __slots__ = ('gender')>>>t = A()>>>t.name = 'Hong'                                      #成功绑定name属性>>>t.score = 98                                         #成功绑定score属性>>>t.gender = 'female'                                  #成功绑定gender属性>>>t.age = 18                                           #age属性绑定失败AttributeError: 'Stu' object has no attribute 'age'

如果子类有__slots__,而父类没有__slots__,则子类的实例并不会受__slots__的影响

class Stu1(object):
    passclass A(Stu1):
    __slots__ = ('gender')

t = A()
t.name = 'Hong'                                        #成功绑定name属性t.score = 98                                           #成功绑定score属性t.gender = 'female'                                    #成功绑定gender属性



作者:三贝_
链接:https://www.jianshu.com/p/d0b32f1b9944


打开App,阅读手记
0人推荐
发表评论
随时随地看视频慕课网APP