Python封装补充

property属性

property实际是setter getter deleter是集合体,并不是一个单独的方法

import math  # 使用的库

class Circle:
    def __init__(self,radius): #定义圆的半径
        self.radius = radius

    @property   # area = property(area)  #
    def area(self):
        return math.pi * self.radius**2  # 在类内部的函数,返回值

    @property
    def peirimeter(self):
        return 2*math.pi*self.radius

c1= Circle(5)  # 对类实例化 产生一个对象
print(c1.radius)  # 打印的是半径的值
print(c1.area)    # 这里相当于是直接使用了一个属性,并没有使用绑定方法
print(c1.peirimeter)

结果:
5
78.53981633974483
31.41592653589793

注意:
c.area是不能赋值的,虽然我们能够当做看似通过属性进行调用,实际是不是的

c1.area = 1

结果是:
AttributeError: can't set attribute
会抛出错误

下面是一个测试:
print(type(Circle.area)) #此时类是不能使用area函数的,结果是class property
print(Circle.area) # 结果是property object at 0x00000000006AB5E8

为什么要用property

将一个函数的类定义成property特性后,对象使用object.name,无法知道这个name是函数计算出来的,这种特性的使用方式遵循统一访问原则

面向对象中有三种类:
public --就是公开类
protected -- 父类
private -- 私有类

但是python的class机制并没有把这三种语法,像C++提供的是set和get方法,Python中是通过property方法提供

class People:
    def __init__(self,name,SEX):
        self.name = name
        self.__sex = SEX  # 进行封装

    @property
    def sex(self):
        return self.__sex  # 这里的返回值是真实的值

p1 = People('aaa','male')
print(p1.name)  #  雷士通过属性的方式进行了调用
print(p1.sex)

结果:
male
aaa

通过属性的的方式是通过property的方式能够object.name的方式,这里就是用来实例化的p1.name p1.sex来是实现

但是这种属性并不是真正的属性,不能通过赋值的方式修改,后期需要进行修改用到了setter方法

setter方法是用了被property装饰器后的函数名命名了一个新的装饰器

class People:
    def __init__(self,name,SEX):
        self.name = name
        self.__sex = SEX  # 进行封装

    @property
    def sex(self):
        return self.__sex  # 这里的返回值是真实的值

    @sex.setter  # 想要对上面得函数属性进行修改
    def sex(self,value):  # value的值是是想要进行修改的参数
        # print('============')  # 测试用
        print(self,value)
        self.__sex = value  # 真实的值是存在与__sex中,赋值给value

p1 = People('aaa','male')   # 实例化
p1.sex = 'femele'  # 这里就是通过类似赋值的方式进行了修改
print(p1.sex)   # 新的查看

结果:
famele

上面的关键是@sex.setter重新命名了一个新的装饰器,装饰器的名字是sex,它的方法是setter。p1.sex = 'femele' 这这执行的时候首先从property下面的sex.__sex,这里存放的是真实的sex的值,所以的赋值只能在这里

通过抛出异常,限制输入的参数必须是字符串

class People:
    def __init__(self,name,SEX):
        self.name = name
        # self.__sex = SEX  # 进行封装
        self.sex = SEX      # 这里直接初始化的时候就会进入@sex.seter

    @property
    def sex(self):
        return self.__sex  # 这里的返回值是真实的值

    @sex.setter  # 想要对上面得函数属性进行修改
    def sex(self,value):  # value的值是是想要进行修改的参数
        # print('============')  # 测试用
        print(self,value)
        if not isinstance(value,str):  # 判断修改的不是字符串,抛出异常
            raise TypeError("male 必须是字符串类型")
        self.__sex = value  # 真实的值是存在与__sex中,赋值给value

p1 = People('aaa','male')   # 实例化
p1.sex = 'femele'
print(p1.sex)

上面的修改是在setter中在修改赋值之前增加了异常处理,如果输入的不是字符串,就会抛出异常,但是这个只是在实例化赋值(p1.sex = 123)时候用才会抛出异常,在类的初始化的时候直接就self.sex=SEX,p1 = People('aaa',123) # 实例化最初的时候就会抛出异常

删除操作

class People:
    def __init__(self,name,SEX):
        self.name = name
        # self.__sex = SEX  # 进行封装
        self.sex = SEX

    @property
    def sex(self):
        return self.__sex  # 这里的返回值是真实的值

    @sex.setter  # 想要对上面得函数属性进行修改
    def sex(self,value):  # value的值是是想要进行修改的参数
        # print('============')  # 测试用
        print(self,value)
        if not isinstance(value,str):  # 判断修改的不是字符串,抛出异常
            raise TypeError("male 必须是字符串类型")
        self.__sex = value  # 真实的值是存在与__sex中,赋值给value

    @sex.deleter
    def sex(self):
        del self.__sex   # 同样删除的是真实存放内容的__sex

p1 = People('aaa','male')
print(p1.sex)
del p1.sex   # 删除p1的sex属性

posted @ 2017-04-23 00:41  hzxPeter  阅读(181)  评论(0编辑  收藏  举报