当在类体内定义好各种属性后,外部是可以随便添加属性的,Python中类如何限制实例的属性?

Python自带了很多定制类,诸如__slots__,__str__

__slots__

__slots__方法是在类创建时,为实例限定属性的

class Student(object):
__slots__=('name','age')
pass >>>s = Student()
>>>s.name='tom'
>>>s.age='age'
>>>s.score= #AttributeError: 'Student' object has no attribute 'score'

当__slots__已经进行限定时,再去给实例绑定没有限定的属性,就会报错。

__slots__作用域仅限当前的类,当父类中有__slots__属性时,子类中并没有限制,当子类中定义__slots__时,子类的实例限制就是子类本身加上父类的__slots__。

__str__和__repr__

如果要把一个类的实例变成 str,就需要实现特殊方法__str__():

class Person(object):
def __init__(self, name, gender):
self.name = name
self.gender = gender
def __str__(self):
return '(Person: %s, %s)' % (self.name, self.gender) >>>p = Person('tom','male')
>>>print(p)
(Person:tom,male)
>>>p #直接打印p __str__不会被调用
<main.Person object at 0x10c941890>

因为 Python 定义了__str__()__repr__()两种方法,__str__()用于显示给用户,而__repr__()用于显示给开发人员:

这个函数,对应repr(object)这个功能。意思是当需要显示一个对象在屏幕上时,将这个对象的属性或者是方法整理成一个可以打印输出的格式。

这个功能与eval也可以对应。打印出的结果直接放到eval里,通常可以获得原来的对象。

>>> from datetime import datetime
>>> t1 = datetime.now()
>>> t1
datetime.datetime(2017, 12, 8, 16, 50, 47, 766613)
>>> print(t1)
2017-12-08 16:50:47.766613
>>> print(repr(t1))
datetime.datetime(2017, 12, 8, 16, 50, 47, 766613)
>>> print(str(t1))
2017-12-08 16:50:47.766613

与此相类似的功能是__str__

比如你有一个People类,当打印时你只希望它打印出人名,那么你可以在__str__函数里写上return self.name , 这样当需要People当作一个字符串时,返回的就是人名。

有一个偷懒的定义__repr__的方法:

class Person(object):
def __init__(self, name, gender):
self.name = name
self.gender = gender
def __str__(self):
return '(Person: %s, %s)' % (self.name, self.gender)
__repr__ = __str__

__len__

如果一个类表现得像一个list,要获取有多少个元素,就得用 len() 函数。

要让 len() 函数工作正常,类必须提供一个特殊方法__len__(),它返回元素的个数。

例如,我们写一个 Students 类,把名字传进去:

class Students(object):
def __init__(self, *args):
self.names = args
def __len__(self):
return len(self.names) >>>s = Student('a','b','c') #只要正确实现了__len__()方法,就可以用len()函数返回Students实例的“长度”
>>>print len(s)
3

__call__

Python所有的函数都是可调用对象。

一个类实例也可以变成一个可调用对象,只需要实现一个特殊方法__call__()

我们把 Person 类变成一个可调用对象:

class Person(object):
def __init__(self, name, sex):
self.name = name
self.sex = sex def __call__(self, friend):
print('My name is %s...' % self.name)
print('My friend is %s...' % friend) >>>p = Person('tom','boy')
>>>p('jerry') #对实例直接调用
My name is tom
My friend is jerry

枚举

定义枚举引入模块enum,枚举类继承的时Enum类

from enum import Enum

class Book(Enum):
a = 1
math = 2
english = 3
linux = 4
c++ = 5

访问枚举有很多方法

>>>Book.a
<Book.a: 1> >>>Book(1)
<Book.a: 1> >>>print(Book.a)
Book.a >>>print(Book(1))
Book.a >>>Book.a.name
'a' >>>Book.a.value
1

如果枚举中成员名有重复的话,就会报错TypeError: Attempted to reuse key: 'a'

成员的值时允许重复的,但是访问时还是第一个成员属性,Python将重复的第二个成员看做第一个的别名

from enum import Enum

class Book(Enum):
a = 1
b = 1 >>>Book.a
<Book.a: 1> >>>Book.b #访问b也变成了打印a属性,b被视为a的别名
<Book.a :1>

如果限制枚举没有重复的话,引入unique模块,unique是一个类装饰器,用来约束值

from enum import Enum,unique

@unique
class Book(Enum):
a = 1
b = 1 >>>print(Book.a.value)
ValueError:dupplicate values found in <enum 'Book'>:b -> a

枚举支持迭代器

>>>for b in Book:
print(b) Book.a、...

成员值重复时,只能拿到第一个成员,若要不管重复,列出所有的,使用__members__方法

from enum import Enum

class Book(Enum):
a = 1
math = 2
english = 3
linux = 4
c++ = 5 for b in Book.__members__.items():
print(b)
#以元组形式打印
('a',<Book.a: 1>)、('math',<Book.math: 2>)、、、

Python面向对象高级编程-__slots__、定制类,枚举的更多相关文章

  1. python 面向对象高级编程

    数据封装.继承和多态只是面向对象程序设计中最基础的3个概念.在Python中,面向对象还有很多高级特性,允许我们写出非常强大的功能. 我们会讨论多重继承.定制类.元类等概念.

  2. python面向对象高级编程

    正常情况下,当我们定义了一个class,创建了一个class的实例后,我们可以给该实例绑定任何属性和方法,这就是动态语言的灵活性.先定义class: >>> class Studen ...

  3. 【Python学习之七】面向对象高级编程——__slots__的使用

    1.Python中的属性和方法的绑定 正常情况下,当我们定义了一个class,创建了一个class的实例后,我们可以给该实例绑定任何属性和方法. (1)首先,定义一个class:  class Stu ...

  4. python面向对象高级:__slots__

    __slots__ 一个在有着数以千计的对象的类的时候节省内存的方法. 在Python中,每个类都有实例属性.默认情况下Python用一个字典来保存一个对象的实例属性.这非常有用,因为它允许我们在运行 ...

  5. Python面向对象高级编程:__slot__(给实例添加方法、属性)

    纲要: 本章总的来说是给实例添加属性,给类添加方法两个主题,以及相应的作用范围.总结如下: 1.给实例添加属性(作用范围:当然是只对当前实例有效): 2.用__slots__限制可以给实例添加的属性( ...

  6. Python面向对象高级编程:@property--把方法变为属性

    为了限制score的范围,可以通过一个set_score()方法来设置成绩,再通过一个get_score()来获取成绩,这样,在set_score()方法里,就可以检查参数: >>> ...

  7. Python面向对象高级编程-@property

    使用@property 在绑定属性时,如果直接把属性暴露出去,虽然写起来简单,但是没法检查参数,导致可以把成绩随便改: >>> class Student(object): pass ...

  8. Python面向对象高级编程-_slots_

    使用_slots_ 正常情况下,当定义一个class,创建一个class的实例后,可以给实例绑定任何属性和方法,这就是动态语言的灵活性.先定义class: >>> class Stu ...

  9. Python:面向对象编程3 定制类(有更新)

    Python:面向对象编程3  定制类(有更新) ⚠️本文主要内容为对Data model相关知识点的提取学习记录.(内容来自文档和部分网页教程案例) ⚠️:这个连接指向<流畅的python&g ...

随机推荐

  1. error: could not create '/System/Library/Frameworks/Python.framework/Versions/2.7/share': Operation not permitted

    参考: Python pip安装模块报错 Mac升级到EI Captain之后pip install 无法使用问题 error: could not create '/System/Library/F ...

  2. C语言 深入学习

    浮点数: x = Mx*2^Ex为一个规格化浮点数,Mx为x的尾数,Ex为x的阶码. 1e-6:表示1 * 10 ^ (-6). 编译时执行: sizeof是运算符(而非函数),在编译时执行,不会导致 ...

  3. .Net ASP.NET 打开指定文件夹

    比如要打开指定的文件夹,而不是弹出对话框 System.Diagnostics.Process.Start(@"D:\"); 这样就打开了D盘,和正常打开D盘是一样的.

  4. Antd-Select组件的深入用法

    一.Antd-Select提供几种类型 最基础版只提供下拉功能的选择器 带搜索功能的下拉选择器 可多选的下拉选择器 可搜索.可多选.可随意输入内容的tag下拉选择器(支持自动分词) 多级联动下拉选择器 ...

  5. 理解 Redis(9) - Publish Subscribe 消息订阅

    在窗口1开通一个名为 redis 的通道: 127.0.0.1:6379> SUBSCRIBE redis Reading messages... (press Ctrl-C to quit) ...

  6. _pvp_gap_aura

    该表可以用于控制区域内平衡,举个例子,当一个区域内超过limitHP的LM玩家个数为10,部落玩家个数为8,则阵营人数差为2,人数少的部落,所有部落玩家获得2层aura光环 zone 区域ID aur ...

  7. PL/SQL Developer过期解决方法

    参考资料: plsql过期解决方法 plsql永久注册码适用个版本 方法一: 1.首先,登陆PL/SQL Developer,PL/SQL Developer要到期了 2.输入指令“regedit”打 ...

  8. SQLServer2008 远程过程调用失败

    今天在连接数据库的时候,发现无法获取到服务器名称,打开SQLServer Configuration Manager,发现SQLServer服务中远程过程调用失败 我装的是VS2017,在网上百度了一 ...

  9. Mybatis中sql语句中的in查询,判断null和size为0的情况

    不严谨的写法,可能会报错:in (),这种情况不符合SQL的语法,导致程序报错. 如果简单只做非空判断,这样也有可能会有问题:本来in一个空列表,应该是没有数据才对,却变成了获取全部数据! 所以一个比 ...

  10. 实现一个键对应多个值的字典(multidict)

    一个字典就是一个键对应一个单值的映射.如果你想要一个键映射多个值,那么你就需要将这多个值放到另外的容器中, 比如列表或者集合里面.比如,你可以像下面这样构造这样的字典: d = { , , ], , ...