python基础----封装、封装与扩展性

   从封装自己的意思去理解,封装就好像是拿来一个麻袋,把小猫,小狗,小王八,还有egon和alex一块儿装进麻袋,而后把麻袋封上口子。但其实这种理解至关片面python

首先咱们要了解编程

要封装什么

你钱包的有多少钱(数据的封装)socket

你的性取向(数据的封装)编程语言

你撒尿的具体功能是怎么实现的(方法的封装)ide

为何要封装

封装数据的主要缘由是:保护隐私(做为男人的你,脸上就写着:我喜欢男人,你惧怕么?)函数

封装方法的主要缘由是:隔离复杂度(快门就是傻瓜相机为傻瓜们提供的方法,该方法将内部复杂的照相功能都隐藏起来了,好比你没必要知道你本身的尿是怎么流出来的,你直接掏出本身的接口就能用尿这个功能)spa

你的身体没有一处不体现着封装的概念:你的身体把膀胱尿道等等这些尿的功能隐藏了起来,而后为你提供一个尿的接口就能够了(接口就是你的。。。,),你总不能把膀胱挂在身体外面,上厕所的时候就跟别人炫耀:hi,man,你瞅个人膀胱,看看我是怎么尿的。还有你的头把你的脑子封装到了脑袋里,而后提供了眼睛这个接口....设计

提示:在编程语言里,对外提供的接口(接口可理解为了一个入口),就是函数,称为接口函数,这与接口的概念还不同,接口表明一组接口函数的集合体。code

封装分为两个层面

封装其实分为两个层面,但不管哪一种层面的封装,都要对外界提供好访问你内部隐藏内容的接口(接口能够理解为入口,有了这个入口,使用者无需且不可以直接访问到内部隐藏的细节,只能走接口,而且咱们能够在接口的实现上附加更多的处理逻辑,从而严格控制使用者的访问)对象

第一个层面的封装(什么都不用作):建立类和对象会分别建立两者的名称空间,咱们只能用类名.或者obj.的方式去访问里面的名字,这自己就是一种封装

>>> r1.nickname
'草丛伦'
>>> Riven.camp
'Noxus'

注意:对于这一层面的封装(隐藏),类名.和实例名.就是访问隐藏属性的接口

第二个层面的封装:类中把某些属性和方法隐藏起来(或者说定义成私有的),只在类的内部使用、外部没法访问,或者留下少许接口(函数)供外部访问。

在python中用双下划线的方式实现隐藏属性(设置成私有的)

类中全部双下划线开头的名称如__x都会自动变造成:_类名__x的形式:

 

class A:
    __N=0 #类的数据属性就应该是共享的,可是语法上是能够把类的数据属性设置成私有的如__N,会变形为_A__N
    def __init__(self):
        self.__X=10 #变形为self._A__X
    def __foo(self): #变形为_A__foo
        print('from A')
    def bar(self):
        self.__foo() #只有在类内部才能够经过__foo的形式访问到.

这种自动变形的特色:

1.类中定义的__x只能在内部使用,如self.__x,引用的就是变形的结果

2.这种变形其实正是针对外部的变形,在外部是没法经过__x这个名字访问到的。

3.在子类定义的__x不会覆盖在父类定义的__x,由于子类中变造成了:_子类名__x,而父类中变造成了:_父类名__x,即双下滑线开头的属性在继承给子类时,子类是没法覆盖的。

注意:对于这一层面的封装(隐藏),咱们须要在类中定义一个函数(接口函数)在它内部访问被隐藏的属性,而后外部就可使用了

 

这种变形须要注意的问题是:

1.这种机制也并无真正意义上限制咱们从外部直接访问属性,知道了类名和属性名就能够拼出名字:_类名__属性,而后就能够访问了,如a._A__N

>>> a=A()
>>> a._A__N
>>> a._A__X
>>> A._A__N

2.变形的过程只在类的定义是发生一次,在定义后的赋值操做,不会变形

3.在继承中,父类若是不想让子类覆盖本身的方法,能够将方法定义为私有的

#正常状况
>>> class A:
...     def fa(self):
...         print('from A')
...     def test(self):
...         self.fa()
... 
>>> class B(A):
...     def fa(self):
...         print('from B')
... 
>>> b=B()
>>> b.test()
from B
#把fa定义成私有的,即__fa
>>> class A:
...     def __fa(self): #在定义时就变形为_A__fa
...         print('from A')
...     def test(self):
...         self.__fa() #只会与本身所在的类为准,即调用_A__fa
... 
>>> class B(A):
...     def __fa(self):
...         print('from B')
... 
>>> b=B()
>>> b.test()
from A

python并不会真的阻止你访问私有的属性,模块也遵循这种约定,若是模块名以单下划线开头,那么from module import *时不能被导入,可是你from module import _private_module依然是能够导入的

其实不少时候你去调用一个模块的功能时会遇到单下划线开头的(socket._socket,sys._home,sys._clear_type_cache),这些都是私有的,原则上是供内部调用的,做为外部的你,独断独行也是能够用的,只不过显得稍微傻逼一点点

python要想与其余编程语言同样,严格控制属性的访问权限,只能借助内置方法如__getattr__,详见面向对象进阶.

 

课堂笔记:

class A:
    __x=1    #_A__x
    def __test(self:   #_A__test
        print('from A')

print(A.__x)
print(A._A__x)
a=A()
print(a._A__x)
print(A.__dict__)
print(A.__dict__)
A._A__test(123)

a=A()
a._A__test()




# __名字,这种语法,只在定义的时候才会有变形的效果,若是类或者对象已经产生了,就不会有变形效果

class B:
    pass

B.__x=1
print(B.__dict__)
print(B.__x)

b=B()
b.__x=1
print(b.__dict__)
print(b.__x)




#在定义阶段就会变形
class A:
    def __fa(self):     #_A__fa
        print('from A')
    def test(self):
        self.__fa()      #self._A__fa

class B(A):
    def __fa(self):    #_B__fa
        print('from B')
b=B()
b.test()




class A:
    def __init__(self):
        self.__x=1
    def tell(self):
        print(self.__x)          # self.__x    在类内部就能够直接用__名字来访问到变形的属性

a=A()
print(a.__dict__)
print(a.__x)
a.tell()
笔记

 

封装与扩展性

封装在于明确区份内外,使得类实现者能够修改封装内的东西而不影响外部调用者的代码;而外部使用用者只知道一个接口(函数),只要接口(函数)名、参数不变,使用者的代码永远无需改变。这就提供一个良好的合做基础——或者说,只要接口这个基础约定不变,则代码改变不足为虑。

#类的设计者
class Room:
    def __init__(self,name,owner,width,length,high):
        self.name=name
        self.owner=owner
        self.__width=width
        self.__length=length
        self.__high=high
    def tell_area(self): #对外提供的接口,隐藏了内部的实现细节,此时咱们想求的是面积
        return self.__width * self.__length
#使用者
>>> r1=Room('卧室','egon',20,20,20)
>>> r1.tell_area() #使用者调用接口tell_area
400
#类的设计者,轻松的扩展了功能,而类的使用者彻底不须要改变本身的代码
class Room:
    def __init__(self,name,owner,width,length,high):
        self.name=name
        self.owner=owner
        self.__width=width
        self.__length=length
        self.__high=high
    def tell_area(self): #对外提供的接口,隐藏内部实现,此时咱们想求的是体积,内部逻辑变了,只需求修该下列一行就能够很简答的实现,并且外部调用感知不到,仍然使用该方法,可是功能已经变了
        return self.__width * self.__length * self.__high

对于仍然在使用tell_area接口的人来讲,根本无需改动本身的代码,就能够用上新功能

>>> r1.tell_area()
8000
相关文章
相关标签/搜索