python之面向对象编制之封装

1.什么是封装

    封:属性对外是隐藏的,单对内是开放的

    装:申请一个名称空间,往里装入一系列名字/属性

2.为什么要封装

    封装数据属性的目的

       首先定义属性的目的就是为了给类外部的使用而使用

            隐藏之后是为了不让外部直接使用,需要内部开辟一个接口

            然后让类外部的使用通过接口间接的操作隐藏的属性

        精髓在于:我们可以在接口之上附加任意逻辑,从而严格控制使用者对属性操作

    封装函数属性

        首先定义属性的目的就是为了给类外部的使用而使用的

                隐藏函数属性是为了不让外部直接使用,需要类的内部开辟一个接口

                 然后在接口内去调用隐藏的功能

        精髓在于:隔离了复杂度        

3.如何封装

    如何隐藏:在属性前面加上__开头

    1.这种隐藏仅仅只是之中语法上的变形操作

    2.这种语法上的变形只在类定义阶段发生一次,因为类体代码仅仅只在类定义阶段检测一次

   3.这种隐藏是对外不对内的,即在类的内部可以直接访问,而在类的外侧无法直接访问

            原因是:

                在类定义阶段,类的体内代码统一发生一次变形

    4.如果不想让子类的方法覆盖父类,可以将该方法前加一个__开头

一.数据属性发封装

例:
# class People:				#隐藏仅仅只是一种语法上的变形操作
#     __country='China' 		#_People__country='China'
#     __n=100 					#_People__n=100
#     def __init__(self,name,age,sex):
#         self.__name=name 		#self._People__name=name
#         self.age=age
#         self.sex=sex
#
#     def eat(self):
#         print('eat.....')
#         print(People.__country) #People._People__country
#         print(self.__name) #self._People__name

# People.eat(123)
# print(People.__country)

# peo1=People('egon',18,'male')
# peo1.eat()
# print(peo1.__name)       	#外面不能调用,属性的名字已经发生改变了

# print(People.__dict__)
# print(People.__country)
# print(People._People__country)

# People.__x=11
# print(People.__dict__)

 

例2:内部属性全部隐藏,设置可以接口查询,和修改的接口,设置操作权限

 

class People:
    def __init__(self,name,age):
        self.__name=name
        self.__age=age

    def tell_info(self):
        print('%s:%s' %(self.__name,self.__age))

    def set_info(self,name,age):
        if type(name) is not str:
            # print('用户名必须为str类型')
            # return
            raise TypeError('用户名必须为str类型')

        if type(age) is not int:
            # print('年龄必须为int类型')
            # return
            raise TypeError('年龄必须为int类型')
        self.__name=name
        self.__age=age

peo1=People('egon',18)
# peo1.name=123                #外部不能直接修改。直接报错
# peo1.age
# peo1.tell_info()

peo1.set_info('egon',19)		#设置一个开放接口,
# peo1.tell_info()

例2.封装加继承属性的查找

# class Foo:
#     def __f1(self): 		#_Foo__f1
#         print('Foo.f1')
#
#     def f2(self):
#         print('Foo.f2')
#         self.__f1() 		#self._Foo__f1
#
# class Bar(Foo):
#     def __f1(self): 		#_Bar__f1
#         print('Bar.f1')
#
# obj=Bar()
# obj.f2()

二.函数属性的封装

property装饰器用于将被装饰的方法伪装成一个数据属性,在使用时可以不加括号而直接引用

class People:

    def __init__(self,name,weight,height):

        self.name=name

        self.weight=weight

        self.height=height

    @property

    def bmi(self):

            return self.weight / (self.height**2)

    peol=People('egon',75,1.8)

    print(peol.bmi)

class People:
    def __init__(self, name, weight, height):
        self.__name = name                      #数据熟悉隐藏+函数属性隐藏
        self.__weight = weight
        self.__height = height
    @property
    def bmi(self):
        return self.__weight / (self.__height ** 2)

peol=People('egon',75,1.8)


print(peol.__dict__)
print(peol.bmi)

 

例,函数封装之 查看、修改、删除,用法:

class People:
    def __init__(self,name):
        self.__name=name

    @property 					# 查看obj.name
    def name(self):
        return '<名字是:%s>' %self.__name

    @name.setter 				#修改obj.name=值
    def name(self,name):
        if type(name) is not str:
            raise TypeError('名字必须是str类型傻叉')
        self.__name=name

    @name.deleter 				#删除del obj.name
    def name(self):
        # raise PermissionError('不让删')       #raise是个异常报错提示命令
        print('不让删除傻叉')
        # del self.__name

peo1=People('egon')
# print(peo1.name)

# print(peo1.name)		   #查看

# peo1.name='EGON'                #修改
# print(peo1.name)

del peo1.name			  #删除 #raise是个异常报错提示命令
        print('不让删除傻叉')
        # del self.__name

peo1=People('egon')
# print(peo1.name)

# print(peo1.name)		   #查看

# peo1.name='EGON'                #修改
# print(peo1.name)

del peo1.name			  #删除

 

2.函数封装:简洁使用方法

class People:
    def __init__(self,name):
        self.__name=name


    def tell_name(self):
        return '<名字是:%s>' %self.__name

    def set_name(self,name):
        if type(name) is not str:
            raise TypeError('名字必须是str类型傻叉')
        self.__name=name

    def del_name(self):
        print('不让删除傻叉')

    name=property(tell_name,set_name,del_name)     #python中简洁使用方法


peo1=People('egon')

print(peo1.name)
peo1.name='EGON'
print(peo1.name)
del peo1.name   name=property(tell_name,set_name,del_name)     #python中简洁使用方法


peo1=People('egon')

print(peo1.name)
peo1.name='EGON'
print(peo1.name)
del peo1.name

 

 

 

posted @ 2018-06-28 17:03  Marcki  阅读(129)  评论(0编辑  收藏  举报