查看原文
其他

Python面向对象之封装(04)

Ahab杂货铺 2019-02-15

  点击上方“Ahab杂货铺”,选择“置顶公众号”

技术分享第一时间送达!



             

前文导读


Python入门都在这了

Python之面向对象(1)

Python面向对象之继承(02)

Python面向对象之多态(03)


知识框架:

1.什么是封装
2.封装的优点与原则

3.私有变量和私有方法
  
4.封装的更多说明

什么是封装

封装是面向对象编程的一大特点,面向对象编程的第一步将属性和方法封装到一个抽象类中,外界使用类创建对象然后让对象调用方法,对象方法的的细节都被封装在类的内部。


封装的优点与原则

【优点】 

将变化隔离; 便于使用;提高复用性; 提高安全性。


【封装原则】

将不需要对外提供的内容都隐藏起来; 把属性都隐藏,提供公共方法对其访问。


私有变量与私有方法

应用场景:在实际开发中对象的某些属性或方法,可能只希望在对象内部被使用,而不希望在外部被访问到。
私有属性就是对象不希望被公开的属性,私有方法就是不希望被公开的方法。

定义方式:在python中用双下划线开头的方式将属性隐藏起来(设置成私有的)


1class Women:
2    def __init__(self,name):
3        self.name=name
4        self.__age=18 
5    def secret(self):
6        #在对象方法内部可以访问私有属性
7        print("%s的年龄是%d"%(self.name,self.__age))
8
9liu=Women("xxx")
10#私有属性或者方法在外不能被直接使用
11print(liu._Women__age)#AttributeError: 'Women' object has no attribute '__age'
12
13在python中没有真正意义的私有 在给属性 方法命名时 实际是对名称做了一些特殊处理
14处理方式  在名称前面加 上_类名    _类名__名称


说明:

1、 _xx 以单下划线开头的表示的是protected类型的变量。即保护类型只能允许其本身与子类进行访问。若内部变量标示,如: 当使用“from M import”时,不会将以一个下划线开头的对象引入 。

2、 __xx 双下划线的表示的是私有类型的变量。只能允许这个类本身进行访问了,连子类也不可以用于命名一个类属性(类变量),调用时名字被改变(在类FooBar内部,__boo变成_FooBar__boo,如self._FooBar__boo)

3、 __xx__定义的是特殊方法。用户控制的命名空间内的变量或是属性,如init , __import__或是file 。只有当文档有说明时使用,不要自己定义这类变量。 (就是说这些是python内部定义的变量名)


python默认的成员函数和成员变量都是公开的,没有像其他类似语言的public,private等关键字修饰.但是可以在变量前面加上两个下划线"_",这样的话函数或变量就变成私有的.这是python的私有变量轧压英文是(private name mangling.) **情况就是当变量被标记为私有后,在变量的前端插入类名,再类名前添加一个下划线"_",即形成了_ClassName__变量名.**


私有方法案例:

1class Test(object):
2    #普通方法
3    def test(self):
4        print("普通方法test")
5    #普通方法
6    def _test1(self):
7        print("普通方法_test1方法")
8    #私有方法
9    def __test2(self):
10        print("私有方法__test2方法")
11
12t = Test()
13t.test()
14t._test1()
15#t.__test2()#调用的时候会报错


私有属性案例:

1class Person(object):
2    def __init__(self, name):
3        self.__name = name
4    def get_name(self):
5        return self.__name
6    def set_name(self, new_name):
7        if len(new_name) >= 5:
8            self.__name = new_name
9        else:
10            print("error:名字长度需要大于或者等于5")
11
12xiaoming = Person("atguigu")
13print(xiaoming.get_name())
14#print(xiaoming.__name)#会报错



在继承中提到过:

子类对象 不能在自己的方法内部直接访问父类的私有属性或私有方法
子类对象 可以通过父类的公有方法间接访问到 私有属性或 私有方法


封装的更多说明

property属性


property是一种特殊的属性,访问它时会执行一段功能(函数)然后返回值


1import math
2class Circle:
3    def __init__(self,radius): #圆的半径radius
4        self.radius=radius
5
6    @property
7    def area(self):
8        return math.pi * self.radius**2 #计算面积
9
10    @property
11    def perimeter(self):
12        return 2*math.pi*self.radius #计算周长
13
14c=Circle(10)
15print(c.radius)
16print(c.area) #可以向访问数据属性一样去访问area,会触发一个函数的执行,动态计算出一个值
17print(c.perimeter) #同上
18'''
19输出结果:
20314.1592653589793
2162.83185307179586
22'''


为什么要用property

将一个类的函数定义成特性以后,对象再去使用的时候obj.name,根本无法察觉自己的name是执行了一个函数然后计算出来的,这种特性的使用方式遵循了统一访问的原则,python并没有在语法上把它们三个内建到自己的class机制中,在C++里一般会将所有的所有的数据都设置为私有的,然后提供set和get方法(接口)去设置和获取,在python中通过property方法可以实现。一个静态属性property本质就是实现了get,set,delete三种方法。

1class Foo:
2    @property
3    def AAA(self):
4        print('get的时候运行我啊')
5
6    @AAA.setter
7    def AAA(self,value):
8        print('set的时候运行我啊')
9
10    @AAA.deleter
11    def AAA(self):
12        print('delete的时候运行我啊')
13
14#只有在属性AAA定义property后才能定义AAA.setter,AAA.deleter
15f1=Foo()
16f1.AAA
17f1.AAA='aaa'
18del f1.AAA


                                                                        

写在最后

python的基础知识已经全部更新完成,后续会将之前介绍的基础知识做个汇总并贴上思维导图,基础知识更新完会来一波python骚操作,介绍python中有意思的库,比如之前的Turtle。请大家继续期待吧




【推荐阅读】

从0开始如何用一个月杀进机器学习比赛Top25%

30行代码实现微信自动回复机器人

【LeetCode】(No.020)有效的括号


欢迎您的点赞和分享

▲长按关注此公众号



    您可能也对以下帖子感兴趣

    文章有问题?点此查看未经处理的缓存