【python基础】封装变量与__slots__用法

封装变量

廖雪峰:https://www.liaoxuefeng.com/wiki/1016959663602400/1017501655757856

当我们设置的属性的时候,直接抛出去给别人用,别人可以随便传值没有限制,当我们当限制属性值时,就不能直接抛出去了。
例如:

class Case:
    def __int__(self,name,age):
        self.name=name
        self.age=age
c=Case()
c.name="sd"
c.age=222

显然,年龄不符合常理。所以我们使用设置属性方法来限制年龄。set_age(value)写入年龄方法,get_age()读取年龄方法。

# 封装

class Case:
    def __int__(self,name,age):
        self.name=name
        #私有属性
        # self.__age=age
        self.set_age(age)

    def get_age(self):
        return self.__age
    def set_age(self,value):
        if value<100:
            self.__age=value
        else:
            raise ValueError("输入年龄有误")

c=Case()
c.name="sd"
c.set_age(4232)
print(c.get_age())
print(c.__dict__)

进阶:上面的调用方法又略显复杂,没有直接用属性这么直接简单。所以需要使用装饰器。装饰器(decorator)可以给函数动态加上功能。Python内置的@property装饰器就是负责把一个方法变成属性调用的:

class Case:
    def __int__(self,name,age):
        self.name=name
        #私有属性
        self.age=age
        # self.set_age(age)
    @property#创建property,负责读取
    def age(self):
        return self.__age
    @age.setter#写入
    def age(self,value):
        if value<100:
            self.__age=value
        else:
            raise ValueError("输入年龄有误")

c=Case()
c.name="sd"
c.age=78
print(c.age)

__slots__用法

slots,定义在类中,作用是为了防止用户输入错误的变量名称。缺点丧失了动态语音给对象添加新的变量。

class Student(object):
    pass
#创建对象
s = Student()

>>> s.name = 'Michael' # 动态给实例绑定一个属性
>>> print(s.name)
Michael

实例绑定一个方法:


>>> def set_age(self, age): # 定义一个函数作为实例方法
...     self.age = age
...
>>> from types import MethodType
>>> s.set_age = MethodType(set_age, s) # 给实例绑定一个方法
>>> s.set_age(25) # 调用实例方法
>>> s.age # 测试结果
25
>>> s2 = Student() # 创建新的实例
>>> s2.set_age(25) # 尝试调用方法
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
AttributeError: 'Student' object has no attribute 'set_age'

为了给所有实例都绑定方法,可以给class绑定方法:


>>> def set_score(self, score):
...     self.score = score
...
>>> Student.set_score = set_score
给class绑定方法后,所有实例均可调用:
>>> s.set_score(100)
>>> s.score
100
>>> s2.set_score(99)
>>> s2.score
99

使用__slots__
但是,如果我们想要限制实例的属性怎么办?比如,只允许对Student实例添加name和age属性。

为了达到限制的目的,Python允许在定义class的时候,定义一个特殊的__slots__变量,来限制该class实例能添加的属性:

class Student(object):
    __slots__ = ('name', 'age') # 用tuple定义允许绑定的属性名称

>>> s = Student() # 创建新的实例
>>> s.name = 'Michael' # 绑定属性'name'
>>> s.age = 25 # 绑定属性'age'
>>> s.score = 99 # 绑定属性'score'
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
AttributeError: 'Student' object has no attribute 'score'

使用__slots__要注意,slots__定义的属性仅对当前类实例起作用,对继承的子类是不起作用的,除非在子类中也定义__slots,这样,子类实例允许定义的属性就是自身的__slots__加上父类的__slots__

>>> class GraduateStudent(Student):
...     pass
...
>>> g = GraduateStudent()
>>> g.score = 9999

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

姑娘别秃头

你的鼓励是为我创作最大的动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值