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

笔记更新于2019年11月27日,
摘要:访问限制;private变量的定义;绑定属性与方法;__slots__的使用


写在前面:为了更好的学习python,博主记录下自己的学习路程。本学习笔记基于廖雪峰的Python教程,如有侵权,请告知删除。欢迎与博主一起学习Pythonヽ( ̄▽ ̄)ノ


文章目录

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

面向对象编程

访问限制

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

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

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

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

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

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

>>>a.name
a
>>>b.__name
AttributeError: '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):
     pass

s = Stu()
s.name = 'Ming'

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

def set_score(self,score):                               #先定义一个函数作为方法
    self.score = score

from types import MethodType                             #加入MethodType模块
s.set_score = MethodType(set_score, s)                   #给实例a绑定set_score方法

调用这个绑定的方法

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

但是该绑定的方法只对实例s有效,对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.score
99
• __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):
    pass

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

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

以上就是本节的全部内容,感谢你的阅读。

下一节内容:15.介绍各种特殊方法的使用,定制类。

有任何问题与想法,欢迎评论与吐槽。

和博主一起学习Python吧( ̄▽ ̄)~*

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,445评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,889评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,047评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,760评论 1 276
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,745评论 5 367
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,638评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,011评论 3 398
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,669评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,923评论 1 299
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,655评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,740评论 1 330
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,406评论 4 320
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,995评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,961评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,197评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,023评论 2 350
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,483评论 2 342

推荐阅读更多精彩内容