属性绑定

  • 在python中可以给类对象动态的绑定属性
  • 但是由于这种特性,随意动态绑定也会带来麻烦,因此可用__slots__来限制可绑定的属性名称
  • __slots__的绑定对于子类是不生效的,只对当前类实例生效
#对于模块Animal.py的说明
'a demo of class type'
#作者
__author__ = 'liyue' class Man(object):
def __init__(self, name):
self.name = name class Woman(object):
#限制绑定的属性名称为name和age,除此之外均不可
__slots__ = ('name', 'age') class PrintInfo(object):
def testMain():
m = Man('zhangsan')
#给m绑定name和age
m.age = 20
m.hight = 180
print(m.age, m.hight) #给w绑定变量,但是hight在绑定列表之外,所以会报错
w = Woman()
w.name = 'lisi'
w.age = 20
w.hight = 180
print(w.name, w.age, w.hight) if __name__ == '__main__':
PrintInfo.testMain()

属性@property

  • 使用@property来定义类的属性,可以对属性进行限制和检查:
#对于模块Animal.py的说明
'a demo of class type'
#作者
__author__ = 'liyue' class Man(object):
#给age赋一个不可变的值
_age = 20
_name = '' #定义age的@property属性
@property
def age(self):
#注意这里的写法
return self._age #定义name的属性
@property
def name(self):
return self._name #定义name的setter方法
@name.setter
def name(self, value):
if not isinstance(value, str):
raise ValueError('name must be a str')
self._name = value class PrintInfo(object):
def testMain():
m = Man()
print(m.age)
#这一句会报错,因为没有对age定义age的setter方法
#m.age = 30 print(m.name)
m.name = 'wangwu'
print(m.name) if __name__ == '__main__':
PrintInfo.testMain()

多继承和Mixin模式

python支持多继承,只需要将继承的类放到定定义中

但是在设计中一个类职责尽量单一,即使有需要增加其他功能,不要通过多重继承来实现。这样的话会增加代码的复杂度。在这里使用Mixin模式通过组合的形式来实现功能

#对于模块Animal.py的说明
'a demo of class type'
#作者
__author__ = 'liyue' #定义主类
class Aniaml(object):
def isAnimal(self):
print( 'i am an animal') #定义不同的功能类,注意,这些功能类职责要单一
#这些类名统一的规则用Mixin作为后缀是为了方便维护和理解,一看就能知道使用了Mixin模式
class FlyMixin(object):
def isFly(self):
print( 'i can fly') class RunMixin(object):
def isRun(self):
print( 'i can run') class PrintInfo(object):
def testMain():
pass #定义实现类,使用了Mixin模式,具备了各种功能
class Bird(Aniaml, RunMixin, FlyMixin):
pass #只继承了基类,所以不支持各种功能
class Man(Aniaml):
pass class PrintInfo(object):
def testMain(self):
print('Bird:')
b = Bird()
b.isAnimal()
b.isFly()
b.isRun() print('man:')
m = Man()
m.isRun() if __name__ == '__main__':
p = PrintInfo()
p.testMain()

类的输出:str__和__repr

在日志或者调试中需要打印类的信息,可以重写__str__方法来实现:


#对于模块Animal.py的说明
'a demo of class type'
#作者
__author__ = 'liyue' #定义主类
class Aniaml(object):
def isAnimal(self):
print( 'i am an animal') #重写Animal类的打印信息,此方法用于输出,对用户
def __str__(self):
return "This is class Animal." #重写了__repr__方法,此方法用于调试输出
def __repr__(self):
return 'This is class Animal.' class PrintInfo(object):
def testMain(self):
a = Aniaml()
#调用了__str__方法
print(a) if __name__ == '__main__':
p = PrintInfo()
p.testMain()

类的迭代器__iter__

iter

Python中类可以在for循环中迭代,类中实现了__iter__()方法即可以在循环中返回迭代对象。和__iter__()方法匹配的还有__next__()方法,二者需要同时完成。

#对于模块Animal.py的说明
'a demo of class type'
#作者
__author__ = 'liyue' class AutoLoop(object):
def __init__(self, maxNum):
#这里定义了变量,所以后面变量都需要用self.var的形式
self.maxNum = maxNum
self._startNum = 0 #定义这个类的迭代器,必须返回自己
def __iter__(self):
return self; #next方法也是必须实现的
def __next__(self):
#循环的执行体
self._startNum += 1
#循环退出的条件,必须有
if self.maxNum < self._startNum:
#退出循环,固定
raise StopIteration()
#每次循环的返回值
return self._startNum if __name__ == '__main__':
#a = AutoLoop(6
for n in AutoLoop(6):
print(n)

类的迭代器-获取指定元素:getitem()

重写_getitem__()方法可以让刚才的例子像list一样通过下标获取指定位置元素的值:

#对于模块Animal.py的说明
'a demo of class type'
#作者
__author__ = 'liyue' class AutoLoop(object):
def __init__(self, maxNum):
#这里定义了变量,所以后面变量都需要用self.var的形式
self.maxNum = maxNum
self._startNum = 0 #定义这个类的迭代器,必须返回自己
def __iter__(self):
return self; #next方法也是必须实现的
def __next__(self):
#循环的执行体
self._startNum += 1
#循环退出的条件,必须有
if self.maxNum < self._startNum:
#退出循环,固定
raise StopIteration()
#每次循环的返回值
return self._startNum #取第n个元素
def __getitem__(self, n):
a, b = 1, 1
for x in range(n):
a, b = b, a + b
return a if __name__ == '__main__':
#a = AutoLoop(6
for n in AutoLoop(6):
print(n) a = AutoLoop(8)
print('取类中的第二个元素的值:%d' % a[2])

这里还可以实现list的切片方法:

#...
def __getitem__(self, n):
if isinstance(n, int): # n是索引
a, b = 1, 1
for x in range(n):
a, b = b, a + b
return a
if isinstance(n, slice): # n是切片
start = n.start
stop = n.stop
if start is None:
start = 0
a, b = 1, 1
L = []
for x in range(stop):
if x >= start:
L.append(a)
a, b = b, a + b
return L
#...

同理__getitem__()的参数检查还要继续完善,才是一个健壮的函数。对于类来说还有__setitem__()和__delitem__()方法可以实现。

枚举类

从Enum派生出我们自己的自定义类,并用@unqire关键字限制重复的枚举对象;

#导入枚举包
from enum import Enum, unique
'a enum class' __author__ = 'liyue' #unique关键字检查重复
@unique
class Wether(Enum):
Sunny = 0
Rain = 1
Snow = 2 if __name__ == '__main__':
w1 = Wether.Sunny
print(w1)
print(w1.value)

Python类的进阶.md的更多相关文章

  1. python 类的进阶

    类的进阶 一 isinstance(obj,cls)和issubclass(sub,super) class Foo: def __init__(self,name): self.name = nam ...

  2. Python类的继承(进阶5)

    转载请标明出处: http://www.cnblogs.com/why168888/p/6411918.html 本文出自:[Edwin博客园] Python类的继承(进阶5) 1. python中什 ...

  3. python 面向对象(进阶篇)

    上一篇<Python 面向对象(初级篇)>文章介绍了面向对象基本知识: 面向对象是一种编程方式,此编程方式的实现是基于对 类 和 对象 的使用 类 是一个模板,模板中包装了多个“函数”供使 ...

  4. python基础——面向对象进阶下

    python基础--面向对象进阶下 1 __setitem__,__getitem,__delitem__ 把对象操作属性模拟成字典的格式 想对比__getattr__(), __setattr__( ...

  5. python基础——面向对象进阶

    python基础--面向对象进阶 1.isinstance(obj,cls)和issubclass(sub,super) isinstance(obj,cls)检查是否obj是否是类 cls 的对象 ...

  6. python面向对象编程进阶

    python面向对象编程进阶 一.isinstance(obj,cls)和issubclass(sub,super) isinstance(obj,cls)检查是否obj是否是类 cls 的对象 1 ...

  7. Python语言学习之Python入门到进阶

    人们常说Python语言简单,编写简单程序时好像也确实如此.但实际上Python绝不简单,它也是一种很复杂的语言,其功能特征非常丰富,能支持多种编程风格,在几乎所有方面都能深度定制.要想用好Pytho ...

  8. 【转】python 面向对象(进阶篇)

    [转]python 面向对象(进阶篇) 上一篇<Python 面向对象(初级篇)>文章介绍了面向对象基本知识: 面向对象是一种编程方式,此编程方式的实现是基于对 类 和 对象 的使用 类 ...

  9. 【转】Python之函数进阶

    [转]Python之函数进阶 本节内容 上一篇中介绍了Python中函数的定义.函数的调用.函数的参数以及变量的作用域等内容,现在来说下函数的一些高级特性: 递归函数 嵌套函数与闭包 匿名函数 高阶函 ...

随机推荐

  1. Android原生和H5交互;Android和H5混合开发;WebView点击H5界面跳转到Android原生界面。

    当时业务的需求是这样的,H5有一个活动商品列表的界面,IOS和Android共用这一个界面,点击商品可以跳转到Android原生的商品详情界面并传递商品ID:  大概就是点击H5界面跳转到Androi ...

  2. 读取txt文件赋值到DataGridView中

    先查看txt是每条信息之间是通过什么分割,我是通过换行符(\n)分割的, 然后再看每一条信息中字段是通过什么分割,我的字段是通过 tab键(\t)分割. 第一步 先获取到txt文件的路径: //获取绝 ...

  3. css实现三角形标

    .iszb{ position: absolute;top: -75px;right:-75px;text-align: center;color: red; width: 150px;height: ...

  4. mysql给查询的结果添加序号

    1.法一: select  (@i:=@i+1)  i,a.url from  base_api_resources a  ,(select   @i:=0)  t2 order by a.id de ...

  5. MySQL数据库相关开发入门

    使用apt-get来进行MYSQL数据库的安装,安装好以后就可以使用数据库了. 命令行键入mysql即可进入(因为数据库初始化的没有密码的):当然为了安全,你最好还是创建一个用户和密码. 当你创建过用 ...

  6. impala 和 kudu 小记

    1. impala(官网) 实时交互SQL大数据查询工具 它提供SQL语义,能查询存储在Hadoop的HDFS和HBase中的PB级大数据. Impala的最大特点也是最大卖点就是它的快速. Impa ...

  7. 3. group_concat与oracle中wm_concat用法一样

    例子如下: select group_concat(rp.ROLE_ID) from eic_right_role_operator rp where rp.OPERATOR_ID = #id#

  8. Nginx 设置负载均衡

    1. 在nginx配置文件目录下另外单独创建一个文件用于管理负载均衡配置,这里起名为 fzjh.conf vim /etc/nginx/fzjh.conf #在文件下添加以下内容 upstream m ...

  9. Linux FACL(文件访问控制列表)

    文件有三种权限 属主权限   属组权限  其他权限 现在有这样一个场景,用户 A 想把文件共享给不是同组内用户 B ,而又不想修改其他权限,这时候 FACL 就起作用了 FACL可以给文件添加一个拓展 ...

  10. centos7下安装python3.7

    记录在2018年最后一个工作日: Linux环境坑爹得要死,环境本身有python2和python3.7两个版本:安装django2的时候,发现默认是python2:把python软连接到python ...