Python第七章__class面向对象高级用法与反射

欢迎加入Linux_Python学习群

 群号:478616847

目录:

  • Python中关于oop的常用术语
  • 类的特殊方法
  • 元类
  • 反射

一、Python中关于oop的常用术语

抽象/实现

抽象指对现实世界问题和实体的本质表现,行为和特征建模,建立一个相关的子集,可以用于 绘程序结构,从而实现这种模型。抽象不仅包括这种模型的数据属性,

还定义了这些数据的接口。对某种抽象的实现就是对此数据及与之相关接口的现实化(realization)。现实化这个过程对于客户 程序应当是透明而且无关的。

封装/接口

封装描述了对数据/信息进行隐藏的观念,它对数据属性提供接口和访问函数。通过任何客户端直接对数据的访问,无视接口,与封装性都是背道而驰的,除非程序

员允许这些操作。作为实现的 一部分,客户端根本就不需要知道在封装之后,数据属性是如何组织的。在Python中,所有的类属性都是公开的,但名字可能被“混

淆”了,以阻止未经授权的访问,但仅此而已,再没有其他预防措施了。这就需要在设计时,对数据提供相应的接口,以免客户程序通过不规范的操作来存取封装的数

据属性。

注意:封装绝不是等于“把不想让别人看到、以后可能修改的东西用private隐藏起来”真正的封装是,经过深入的思考,做出良好的抽象,给出“完整且最小”

的接口,并使得内部细节可以对外透明(注意:对外透明的意思是外部调用者可以顺利的得到自己想要的任何功能,完全意识不到内部细节的存在)

合成

合成扩充了对类的 述,使得多个不同的类合成为一个大的类,来解决现实问题。合成 述了 一个异常复杂的系统,比如一个类由其它类组成,更小的组件也可能是

其它的类,数据属性及行为, 所有这些合在一起,彼此是“有一个”的关系。

派生/继承/继承结构

派生描述了子类衍生出新的特性,新类保留已存类类型中所有需要的数据和行为,但允许修改或者其它的自定义操作,都不会修改原类的定义。

继承描述了子类属性从祖先类继承这样一种方式

继承结构表示多“代”派生,可以述成一个“族谱”,连续的子类,与祖先类都有关系。

泛化/特化

基于继承

泛化表示所有子类与其父类及祖先类有一样的特点。

特化描述所有子类的自定义,也就是,什么属性让它与其祖先类不同。

多态与多态性

多态指的是同一种事物的多种状态:水这种事物有多种不同的状态:冰,水蒸气

多态性的概念指出了对象如何通过他们共同的属性和动作来操作及访问,而不需考虑他们具体的类。

冰,水蒸气,都继承于水,它们都有一个同名的方法就是变成云,但是冰.变云(),与水蒸气.变云()是截然不同的过程,虽然调用的方法都一样

自省/反射

自省也称作反射,这个性质展示了某对象是如何在运行期取得自身信息的。如果传一个对象给你,你可以查出它有什么能力,这是一项强大的特性。如果

Python不支持某种形式的自省功能,dir和type内建函数,将很难正常工作。还有那些特殊属性,像__dict__,__name__及__doc__

二、类的特殊方法

1、isinstance和issubclass

isinstance(obj,cls)检查是否obj是否是类 cls 的对象

 class Foo(object):
pass
obj = Foo()
print(isinstance(obj, Foo))

isinstance

issubclass(sub, super)检查sub类是否是 super 类的派生类

 class Foo(object):
pass
class Test(Foo):
pass
print(issubclass(Test, Foo))

issubclass

2、__setattr__,__delattr__,__getattr__

__setattr__只要添加和修改属性就会触发它的运行,可以利用它修改和添加属性

 class Foo:
def __init__(self,y):
self.y=y
def __setattr__(self, key, value):
print('我是 setattr')
# self.key=value #这就无限递归了
self.__dict__[key]=value #所以应该利用dict def func():
print("from aaa") #__setattr__添加/修改属性会触发它的执行
f1=Foo(10) #在init构造函数中会触发
print(f1.__dict__) #输出未赋值的对象dict
f1.a=func #赋值
print(f1.__dict__)#输出赋值完成的对象dict
f1.a() #执行

setattr

__delattr__删除属性的时候会触发它,可以利用它来限制哪些属性不能被删除

 class Foo:
x=1
def __init__(self,y):
self.y=y def __delattr__(self, item):
print('触发 delattr')
if item == "y":return#可以做判断保证一些属性不能被删除
self.__dict__.pop(item) #实例化
f1=Foo(10) #__delattr__删除属性的时候会触发
f1.__dict__['a']=10#我们可以直接修改属性字典,来完成添加/修改属性的操作
print(f1.__dict__)
del f1.y
print(f1.__dict__)

delattr

__getattr__只有在使用点调用不存在的属性时候才会触发

 class Foo:
x=1
def __init__(self,y):
self.y=y def __getattr__(self, item):
print('触发 getattr')
if item == 'age':
return 26
else:
print("你访问的属性不存在") #__setattr__添加/修改属性会触发它的执行
f1=Foo(10)
print(f1.y) #__getattr__只有在使用点调用不存在的属性时候才会触发
print(f1.age) print(f1.test)

getattr

3、__getattribute__

这个方法表示只要用点调用就会触发,当_getattribute__与__getattr__同时存在,只会执行__getattrbute__,除非__getattribute__在执行过程中抛出

异常AttributeError

 class Foo:
def __init__(self,x):
self.x=x def __getattr__(self, item):
print('执行的是我')
# return self.__dict__[item]
def __getattribute__(self, item):
print('不管是否存在,我都会执行')
raise AttributeError() f1=Foo(10)
f1.x #存在触发 f1.test #不存在触发
#当__getattribute__与__getattr__同时存在,只会执行__getattrbute__,除非__getattribute__在执行过程中抛出异常AttributeError

getattribute

4、__getitem__,__setitem__,__delitem__

用于索引操作,如字典。以上分别表示获取、设置、删除数据

 class Foo(object):
def __init__(self,name):
self.name = name
def __getitem__(self, key):
print('getitem',key) def __setitem__(self, key, value):
print('setitem',key,value)
self.__dict__[key]=value def __delitem__(self, key):
print('delitem',key)
# self.__dict__.pop(key) obj = Foo("FFF") print(obj.__dict__) result = obj['k1'] # 获取自动触发执行 __getitem__
obj['name'] = 'test' # 设置自动触发执行 __setitem__
del obj['k1'] # 删除自动触发执行 __delitem__ print(obj.__dict__)

setitem,getitem,delitem

5、__str____repr__

__str__,在打印对象时触发

 class Foo:
def __str__(self):
print("hehe")
return 'test' obj = Foo()
print(obj)

str

__repr__,事实上repr跟str做了一件事,唯一的区别就是,python说str输出的是对人类友好,repr对机器友好,记不记得第一章中的字符串格式化,就提到了

  %s 字符串 (采用str()的显示) %r 字符串 (采用repr()的显示)

 class Foo(object):
def __str__(self):
return "str"
def __repr__(self):
return "repr" obj = Foo()
print("%s" %obj)
print("%r" %obj)

repr

6、__slots__

在python新式类中,从object继承下来的类有一个变量是__slots__,默认情况下每个类都会有一个dict,slots的作用是阻止在实例化类时为实例分配dict,由

__slots__管,目的是节省内存。

 class Foo:
__slots__=['name','age'] f1=Foo()
f1.name='test'
f1.age=22
print(f1.__slots__)
#这句会报错,原因是没有__dict__
#print(f1.__dict__) print(Foo.__dict__)

slots

7、__iter__和__next__

这两个主要是实现迭代器协议的,下面的例子中可以看出,首先执行__iter__,然后返回iter(),iter表示如果传递了第二个参数,则第一个参数必须是一个可调用

的对象(如,函数)。此时,iter创建了一个迭代器对象,用for循环这个迭代器时,都会调用第一个参数。如果__next__的返回值等于第二个值,则停止,否则

返回下一个值,也可以不传入第二个参数,直接返回self,那么在for循环和使用next()的时候就会一直迭代。

 class Foo(object):
def __init__(self):
self.x = ["a","b","c"]
self.a =1
def __iter__(self):
print("__iter__")
return iter(self.__next__,4) def __next__(self):
print("__next__")
self.a+=1
return self.a itr = Foo()
for i in itr:
print(i)

__iter__,__next__

 class Foo:
def __init__(self,x):
self.x=x
def __iter__(self):
return self def __next__(self):
if self.x == 0:
raise StopIteration
self.x-=1
return self.x f1=Foo(5)
print(next(f1))
print(next(f1))
for i in f1:
print(i)

__iter__,__next__(2)

8、__doc__

返回类的描述信息,这个属性无法继承给子类

 class Foo:
"""
Hello 我是描述信息
"""
def __init__(self):
pass def test(self):
pass print(Foo.__doc__)

doc

9、__module__和__class__

__module__表示当前操作的对象在哪个模块,__class__表示当前操作的对象属于哪个类

 #!/usr/bin/env python

 class Foo(object):

     def __init__(self):
self.name = "test"

test2

 from test2 import Foo

 obj = Foo()

 print(obj.__module__)
print(obj.__class__)

module,class

10、__del__

__del__当对象在内存中被释放时,自动触发执行,也就是说del 删除对象和运行结束释放对象的时候都会执行

 class Foo:

     def __del__(self):
print('执行我啦') obj = Foo()
obj2 = Foo() del obj
input(">>>>>>")

del

11、__enter__和__exit__

我们可以这样操作文件 with open(‘a.txt’) as f ,这叫做上下文管理协议,即with语句,为了让一个对象兼容with语句,必须在这个对象的类中声明__enter__

和__exit__方法。

 class Open:
def __init__(self,name):
self.name=name def __enter__(self):
print('出现with语句,对象的__enter__被触发,有返回值则赋值给as声明的变量')
return "test"
def __exit__(self, exc_type, exc_val, exc_tb):
print('with中代码块执行完毕时执行我啊') with Open('a.txt') as f:
print('f:',f)

enter,exit

12、__call__

__call__表示生成的对象执行对象()就会触发 call 方法

 class Foo:

     def __call__(self, *args, **kwargs):
print("__call__") obj = Foo()
obj()

call

上述方法中其实很多都不常用,只需要了解一下即可

三、元类(metaclass)

什么是元类,在Python中一切皆对象,类本身也是对象当使用关键字class的时候,python解释器在加载class的时候就会创建一个对象(这里的对象指的是类而非类

的实例),我们可以使用type查看元类是谁

 class Foo:
pass
print(type(Foo))

查看元类

什么是元类?

元类就是类的祖宗,也就是类的模板

在这个模板中,定义了python是如何创建类的,并且可以用这个模板创建一个类

 def func(self,name):
print(self)
print('from func:',name) Foo=type('Foo',(object,),{'func':func,'x':1}) obj = Foo()
obj.func("test")
print(obj.x)

利用type元类创建类

  但是感觉上面好像少点什么,是不是没有构造方法

 def func(self,):
print(self)
print('from func:',self.name) def init(self,name):
self.name = name
Foo=type('Foo',(object,),{'func':func,'x':1,"__init__":init}) obj = Foo("test")
obj.func()
print(obj.x)

添加构造方法

  

  类的创建过程

  好到此为止已经可以使用type元类创建一个类了,但是这个类是如果创建的,type创建类过程是什么样子的?

  首先要了解为什么会执行 __init__构造方法,__init__之所以被执行,是因为有__new__来去调用它,__new__里面干了就是实例化的过程,当我们创建一个__new__

  后就直接执行__new__,我们自己定义的__new__方法中没有实例化的过程所以无法实例化,也不能调用 self.name

 class Foo(object):
def __init__(self,name):
self.name = name
print("__init__") def __new__(cls, *args, **kwargs):
print("__new__") f = Foo("test") #报错
#print(f.name)

__new__

那么如果用自定义的new进行实例化呢?

首先执行 object.__new__(cls)  (cls 就是类的本身),然后触发 init 之后生成一块内存地址,然后通过return赋值给实例 f我们可以打印一下 object.__new__(cls)

的内存地址,和 f的内存地址,它们的地址是一样的。

 class Foo(object):
def __init__(self,name):
self.name = name
print("__init__") def __new__(cls, *args, **kwargs):
print("__new__",*args, **kwargs)
return object.__new__(cls) f = Foo("test") print(f.name)

自定义new进行实例化

到现在我们知道了通过 new 调用 init 然后生成内存空间,赋值给实例,那么这个过程我们是可以控制的,看下面的例子

1、首先通过 class Foo (metaclass=MyType)  metaclass指定元类是MyType

2、然后Mytype 创造出Foo这个类对象,创造过程就是先执行 MyType 的 new 然后执行 init

3、然后 Foo(“test”) 就执行了 MyType 的 call方法,在call方法中来定义,Foo实例化的过程,(可以把注释的地方解开)

4、这个过程首先通过 Foo的new方法创造实例的内存空间,又执行了obj.age = 22 进行了赋值(这里为了体现f.age),然后执行Foo的 init 进行实例化

5、最后通过return 把创造出来的实例的内存空间,返回给 f,这时f就是Foo的实例,(如果不明白可以加断点走一变)

 class MyType(type):
# def __init__(self,*args,**kwargs):
# print("MyType __init__",self,*args,**kwargs) def __call__(self, *args, **kwargs):
print("MyType __call__")
obj = self.__new__(self)
self.__init__(obj, *args, **kwargs)
obj.age = 25
return obj # def __new__(cls, *args, **kwargs):
# print("MyType __new__",)
# return type.__new__(cls,*args, **kwargs) class Foo(metaclass=MyType):
def __init__(self,name):
self.name = name
print("Foo __init__") f = Foo("test")
print("查看Foo的类",type(Foo))
print(f.name)
print(f.age)

创建类的流程

到此我们演示了 类的创建过程,具体有什么用,到后期会一一揭开  

四、反射

反射的概念是由Smith在1982年首次提出的,主要是指程序可以访问、检测和修改它本身状态或行为的一种能力(自省)。这一概念的提出很快引发了计算机科学

领域关于应用反射性的研究。它首先被程序语言的设计领域所采用,并在Lisp和面向对象方面取得了成绩。

反射就是通过类名获得类的实例对象(一切皆对象,所以只要是对象就能用反射),通过方法名得到方法,实现调用,在python中反射功能是由四个内置函数提供的

  • hasattr 是否含有某成员
  • getattr 获取成员
  • setattr 设置成员
  • delattr 删除成员
 class Foo(object):

     def __init__(self):
self.name = 'test' def func(self):
print("func")
return 'func' obj = Foo() #### 检查是否含有成员 ####
print(hasattr(obj, 'name'))
print(hasattr(obj, 'aaa')) #### 获取成员 ####
print(getattr(obj, 'name'))
getattr(obj, 'func')() #### 设置成员 ####
setattr(obj, 'age', 18) #### 删除成员 ####
print(obj.__dict__)
#注意哦,这里不能删除 func 这个方法,因为在 obj中没有这个方法,这个方法是在 Foo中的
delattr(obj, 'name')
print(obj.__dict__)

反射的用法

  

作者:北京小远
出处:http://www.cnblogs.com/bj-xy/
本文版权归作者和博客园共有,欢迎转载,但未经作者同意必须在文章页面明显位置给出原文连接,否则保留追究法律责任的权利。

简学Python第七章__class面向对象高级用法与反射的更多相关文章

  1. 简学Python第六章__class面向对象编程与异常处理

    Python第六章__class面向对象编程与异常处理 欢迎加入Linux_Python学习群  群号:478616847 目录: 面向对象的程序设计 类和对象 封装 继承与派生 多态与多态性 特性p ...

  2. 简学Python第四章__装饰器、迭代器、列表生成式

    Python第四章__装饰器.迭代器 欢迎加入Linux_Python学习群  群号:478616847 目录: 列表生成式 生成器 迭代器 单层装饰器(无参) 多层装饰器(有参) 冒泡算法 代码开发 ...

  3. 简学Python第三章__函数式编程、递归、内置函数

    #cnblogs_post_body h2 { background: linear-gradient(to bottom, #18c0ff 0%,#0c7eff 100%); color: #fff ...

  4. 简学Python第五章__模块介绍,常用内置模块

    Python第五章__模块介绍,常用内置模块 欢迎加入Linux_Python学习群  群号:478616847 目录: 模块与导入介绍 包的介绍 time &datetime模块 rando ...

  5. 从零开始学Python第七周:面向对象进阶(需修改)

    一,类的属性 (1)示例 通过属性获取已经创建对象的个数 class Plane: pCount = 0 #类属性 def __init__(self,name,category): self.nam ...

  6. 简学Python第二章__巧学数据结构文件操作

    #cnblogs_post_body h2 { background: linear-gradient(to bottom, #18c0ff 0%,#0c7eff 100%); color: #fff ...

  7. react第七单元(组件的高级用法-组件的组合(children的用法)-高阶组件-封装组件)

    第七单元(组件的高级用法-组件的组合(children的用法)-高阶组件-封装组件) #受控组件 简而言之,就是受到状态state控制的表单,表单的值改变则state值也改变,受控组件必须要搭配onc ...

  8. 简学Python第一章__进入PY的世界

    #cnblogs_post_body h2 { background: linear-gradient(to bottom, #18c0ff 0%,#0c7eff 100%); color: #fff ...

  9. Python第七章-面向对象高级

    面向对象高级 一. 特性 特性是指的property. property这个词的翻译一直都有问题, 很多人把它翻译为属性, 其实是不恰当和不准确的. 在这里翻译成特性是为了和属性区别开来. 属性是指的 ...

随机推荐

  1. 在 Windows 上安装 Hadoop 教程(转)

    在 Windows 上安装 Hadoop 教程 一见 2010.1.6 www.hadoopor.com/hadoopor@foxmail.com 1. 安装 JDK 不建议只安装 JRE,而是建议直 ...

  2. Vimium -为键盘而生

    The hacker's browser. 作为一个Chrome的忠实使用者,从开发人员工具到谷歌的扩展程序(extensions)[插件],这些都在无形之中提高我们的工作效率. N年前的一天,看到了 ...

  3. [UWP]附加属性2:实现一个Canvas

    5. 附加属性实践:自定义Canvas 附加属性在UWP中是一个十分重要的组成部分,很多功能都依赖于附加属性实现,典型的例子是常用的Grid和Canvas.通常附加属性有三个使用场景:插入属性.触发行 ...

  4. jQuery源码学习:Deferred Object

    本文所有讨论均基于jQuery版本3.1.1,官网http://jquery.com/. 一.Deferred Object 1. 简介和创建 详见API:http://api.jquery.com/ ...

  5. index_merge引发的死锁排查

    概述 前几天排查了一个死锁问题,最开始百思不得其解,因为发生死锁的两个事务是单语句事务,语句类型相同(where属性列相同,仅值不同),而且语句都走了相同的索引,但最终确实发生了死锁.通过定位排查发现 ...

  6. 发布自己的Angular2库初探

    从去年年底开始使用ng2,遇到并解决或被虐了一些问题点,对其各种新特性与开发模式感觉还算舒服.还有的一个感想就是,要使用ng2还得先学习不少其他东西,比如TypeScript语法,比如ES6新特性,还 ...

  7. 教你如何一步步将项目部署到Github

    注册Github账号有半年多的时间,却一直不知道如何将自己做好的项目部署到Github中.看了网上许多的教程,要么一开始就来Git命令行,要么直接就来一堆术语,很少能够真正说中要点,解决我们的烦恼. ...

  8. iOS 用Swipe手势和动画实现循环播放图片

    主要想法 添加3个ImageView展示图片,实现图片的无限循环. 使用Swipe手势识别用户向右或向左滑动图片. 使用CATransition给ImageView.layer添加动画,展示图片更换的 ...

  9. 360浏览器和猎豹浏览器重定向后丢失get参数

    1.场景: step1: 用户请求项目 http://localhost:8080/myProject step2: 我们的服务器将请求重定向到一个静态页面,并拼接上部分参数,如: <%@ pa ...

  10. mysql 添加登录用户

    一, 创建用户: 命令:CREATE USER 'username'@'host' IDENTIFIED BY 'password'; 说明:username - 你将创建的用户名, host - 指 ...