前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >与面试官谈笑风生 | Python面向对象之访问控制

与面试官谈笑风生 | Python面向对象之访问控制

作者头像
simpleapples
发布2018-10-18 14:56:12
3830
发布2018-10-18 14:56:12
举报
文章被收录于专栏:Python私房菜Python私房菜

Python从设计之初就是一门面向对象的语言,面向对象思想的第一个要素就是封装。所谓封装,通俗的讲就是类中的属性和方法,分为公有和私有,公有可以被外界访问,私有不能被外界访问,这就是封装中最关键的概念——访问控制。

访问控制有三种级别:私有、受保护、公有

私有(Private):只有类自身可以访问

受保护(Protected):只有类自身和子类可以访问

公有(Public):任何类都可以访问

由于Python不像Java,有访问控制符(private / public / protected),所以Python的访问控制也是容易被应聘者忽视和搞错的。

公有(Public)

在Python的类中,默认情况下定义的属性都是公有的。

代码语言:javascript
复制
 1class Foo(object):
 2    bar = 123
 3
 4    def __init__(self, bob):
 5        self.bob = bob
 6
 7print(Foo.bar)  # 123
 8
 9foo = Foo(456)
10print(foo.bob)  # 456

上面类Foo中的属性就是类属性,__init__方法中定义的bob是实例属性,bar和都是公有的属性,外部可以访问,分别print类中的bar和实例中的bob,输出了对应的值。

受保护(Protected)

在Python中定义一个受保护的属性,只需要在其名字前加一个下划线_,我们将Foo方法中的bob和bar改为_bob_bar,他们就变成了受保护的属性了,代码如下:

代码语言:javascript
复制
 1class Foo(object):
 2    _bar = 123
 3
 4    def __init__(self, bob):
 5        self._bob = bob
 6
 7
 8class Son(Foo):
 9
10    def print_bob(self):
11        print(self._bob)
12
13    @classmethod
14    def print_bar(cls):
15        print(cls._bar)
16
17
18Son.print_bar()  # 123
19
20son = Son(456)
21son.print_bob()  # 456

定义一个类Son继承自Foo,由于受保护的对象只能在类的内部和子类中被访问,不能直接调用print(Son._bar)print(son._bob)来输出这两个属性的值,所以定义了print_barprint_bob方法,实现在子类中输出,这段代码也正常的输出了_bar_bob的值。

接下来,试着反向验证一下,在类的外部,能不能访问其属性,将上面代码的输出部分修改如下:

代码语言:javascript
复制
1print(Son._bar)  # 123
2
3son = Son(456)
4print(son._bob)  # 456

(假装)惊讶的发现,竟然没有报错,也输出了正确的值。

Python中用加下划线来定义受保护变量,是一种约定的规范,而不是语言层面真的实现了访问控制,所以,我们定义的保护变量,依然可以在外部被访问到(这是个feature,不是bug)。

私有(private)

Python定义私有属性,需要在属性名前加两个下划线__,把上面的代码修改一下,运行一下会发现下面的代码中的任何一个print都会报错的。

代码语言:javascript
复制
 1class Foo(object):
 2    __bar = 123
 3
 4    def __init__(self, bob):
 5        self.__bob = bob
 6
 7
 8class Son(Foo):
 9
10    def print_bob(self):
11        print(self.__bob)  # Error
12
13    @classmethod
14    def print_bar(cls):
15        print(cls.__bar)  # Error
16
17
18print(Son.__bar)  # Error
19
20son = Son(456)
21print(son._bob)  # Error

深入一下——私有属性真的就访问不到了吗?

要了解私有属性是否真的访问不到,需要从Python是如何实现私有属性入手。CPython中,会把双下划线的属性变为_ClassName__PropertyName的形式,用代码演示一下:

代码语言:javascript
复制
1class Foo(object):
2    __bar = 123
3
4
5print(Foo._Foo__bar)  # 123

运行一下可以知道,正常输出了__bar的值,但是不推荐这样去访问私有属性,因为不同的Python解释器对于私有属性的处理不一样。

特例

使用双下划线定义私有属性,有一种特殊情况,当属性后也有两个下划线的时候,这个属性会被Python解释器当做魔术方法,从而不做私有处理。

代码语言:javascript
复制
1class Foo(object):
2    __bar__ = 123
3
4
5print(Foo.__bar__)  # 123

上面代码输出了123,证明Python解释器并没有把__bar__当做私有属性。当定义私有属性时,需要注意名字最后最多只能有一个下划线。

另一个特例

假如定义的属性名就叫__呢?不妨直接试一下:

代码语言:javascript
复制
1class Foo(object):
2    __ = 123
3
4
5print(Foo.__)  # 123

可以发现名字叫__的属性也不会被认为是私有属性,名字是多个下划线的属性也不是私有属性(比如_______)。

函数的访问控制

前面主要介绍了属性的访问控制,在Python中函数是一等公民,所谓一等公民,就是函数可以像变量一样使用,所以函数的访问控制和属性一样,一样应用上面的规则。

本文参与 腾讯云自媒体分享计划,分享自微信公众号。
原始发表:2018-04-02,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 Python私房菜 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档