对于python 3.x与python2.x中新型类的继承特性总结
(1)一般性继承特性
"""
该文件对于python 3.x 及python 2.x的New-style 类的一般性继承特性进行了说明和测试。
(1)实例的继承特性:搜寻实例,类,超类:
a.搜寻该实例的__class__(即该实例的类)的__mro__中的所有类的__dict__
b.如果在步骤a中,第一个找到的是相应的数据描述符,则调用并退出(找到后仅判断是否为描述符,不再继续往后找)
c.否则,如果不是数据描述符,返回实例的__dict__中的值
d.否则,实例__dict__没有,调用在a中找到的非数据描述符,或者返回在a中找到的值
(2)类的继承特性:搜寻类,超类,元类:
a.搜寻该类的__class__(即该类的元类)的__mro__中所有元类的__dict__
b.如果在步骤a中,第一个找到的相应的数据描述符,则调用并退出(找到后仅判断是否为描述符,不再继续往后找)
c.否则,如果不是数据描述符,调用或返回在该类自己的__mro__中所有类的__dict__中的第一个找到的描述符或值
d.否则,调用在a中找到的非数据描述符,或者返回在a中找到的值 对于赋值继承,实例继承从上述a到c,不过b步骤改为__set__方法,c步骤为在继承里存储
类继承与一般性继承特性一致,只有c步骤改为停止,储存
"""
class desc: #数据描述符
def __get__(self,instance,owner):
return ('datadesc getting!')
def __set__(self,instance,value):
print ('datadesc setting!',value)
class Nondatadesc: #非数据描述符
def __get__(self,instance,owner):
return ('Nondatadesc getting')
class a(type):
data1=desc() #类继承,第一个找到数据描述符
data2='only located in metaclass a' #类继承,因为第一个找到的是该处,而该处又非数据描述符,而该类基类__dict__中又无,故返回该值
data10=Nondatadesc() #类继承,找到非数据描述符,同上,最终返回该值
data11=Nondatadesc() #类继承,第一个找到该处,但非数据描述符,在step c中,找到data11,故此处被截断
class b:
data1='b1' #类继承,其值被描述符截断
data3=desc() #类继承,超类中找到的值
def __init__(self):
self.data8='cannot be found' #实例继承找不到该值,因为不在实例类的超类的__dict__中
data5='b5' #实例继承,被前面的d中的描述符截断
data7='found in superclass b' #实例继承,返回仅能在b中找到的值
data6=desc() #实例继承,于其类的超类中找到的数据描述符
data9='being intercepted by a.data9' #实例继承,a中无描述符
data11='Fetched before data11 in class a' #类继承,step c 返回该值,截断元类
data12=Nondatadesc() #实例继承中,最终返回非数据描述符
class d:
data1='d1'
data6='being found prior to b,so interceptting b"s desc' #实例继承中,第一次找到,但不是desc,进入step c,实例中没有data6,故最终返回该值
data5=desc() #实例继承中,第一次找到的data5是描述符
data3='First meet d,so interceptting b"s desc' #类继承中,第一次找到的daa3,不是 class c(d,b,metaclass=a):
data1='c1' #类继承中,其值被描述符截断
data4='located in class c' #实例继承,于其类中找到值
def __init__(self):
self.data9='instance dict' #实例继承步骤c中找到该值
# self.data5='instance data5' print('Test Starting'.center(50,'-'))
import sys
print('sys.verson=',sys.version)
print('实例继承测试'.center(50,'*'))
x=c()
print('x.data5',x.data5)
print('x.data9',x.data9)
print('x.data7',x.data7)
print('x.data6',x.data6)
print('x.data12',x.data12)
x.data5=1 #触发__set__方法
x.data9=2
# print('set x.data5=1->',x.data5)
# x.data9=2
# print('set x.data9=2->',x.data9)
try:
x.data8
except AttributeError:
print('impossible fetch')
print('实例搜索路径'.center(50,'*'))
for cls in x.__class__.__mro__:
for attr in cls.__dict__:
if not attr.startswith('__'):
print('class %s--> attr %s'%(cls.__name__,attr))
print('类继承测试'.center(50,'*'))
print('c.data1',c.data1)
print('c.data3',c.data3)
print('c.data2',c.data2)
print('c.data10',c.data10)
print('c.data11',c.data11)
c.data1=1 #触发__set__方法
c.data3=3
运行后:
------------------Test Starting-------------------
sys.verson= 3.6.8 (tags/v3.6.8:3c6b436a57, Dec 24 2018, 00:16:47) [MSC v.1916 64 bit (AMD64)]
**********************实例继承测试**********************
x.data5 datadesc getting!
x.data9 instance dict
x.data7 found in superclass b
x.data6 being found prior to b,so interceptting b"s desc
x.data12 Nondatadesc getting
datadesc setting! 1
impossible fetch
**********************实例搜索路径**********************
class c--> attr data1
class c--> attr data4
class d--> attr data1
class d--> attr data6
class d--> attr data5
class d--> attr data3
class b--> attr data1
class b--> attr data3
class b--> attr data5
class b--> attr data7
class b--> attr data6
class b--> attr data9
class b--> attr data11
class b--> attr data12
**********************类继承测试***********************
c.data1 datadesc getting!
c.data3 First meet d,so interceptting b"s desc
c.data2 only located in metaclass a
c.data10 Nondatadesc getting
c.data11 Fetched before data11 in class a
datadesc setting! 1
(2)built-ins特殊情况
"""
builtin 的继承特性:
(1)对于实例继承:若是显式调用,则首先搜索实例,否则搜索该实例的类,否则搜索实例类的超类;
若是隐式调用,则跳过实例,首先搜索该实例的类,否则搜索该实例类的超类。
(2)对于类继承:若是显示调用,则首先搜索该类,否则搜索该类的超类;
若是隐式调用,则跳过该类,首先搜索该类的元类(metaclass),否则搜索该元类的超类。
注意到2点:一是实例和类的共同点,即对于显示调用,都是从自身层搜索,然后搜索自身的继承层(实例继承层是类,
类继承层是超类;而隐式调用首先从自身的创建层(实例创建层是类,类创建层是元类),否则再搜索创建层的继承层。
"""
class a(type):
def __str__(self):
return 'a'
class b(type):
pass
class c(a):
def __str__(self):
return 'c'
class d(metaclass=a):
pass
class e(metaclass=a):
def __str__(self):
return 'e'
class f(metaclass=b):
def __str__(self):
return 'f'
class g(c):
pass
class h(d,e,metaclass=g):pass
print('实例显式继承'.center(50,'*'))
x=h()
print('x.__str__:=>',x.__str__()) #实例无_str__,类无,超类d无,超类e有,返回'e'
x.__str__=lambda :'lambda'
print('Having set instance x"s __str__')
print('x.__str__:=>',x.__str__()) #实例有__str__
print('实例隐式继承'.center(50,'*'))
print('str(x)',str(x)) #从类开始搜索,然后搜类的超类,找到e
print('类显式继承'.center(50,'*'))
print('h.__str__(h)',h.__str__(h)) #从类开始搜索,然后搜索类的超类,找到e
print('类隐式继承'.center(50,'*'))
print('str(h)',str(h)) #从元类搜索,然后搜索元类的超类,找到c
**********************实例显式继承**********************
x.__str__:=> e
Having set instance x"s __str__
x.__str__:=> lambda
**********************实例隐式继承**********************
str(x) e
**********************类显式继承***********************
h.__str__(h) e
**********************类隐式继承***********************
str(h) c
[Finished in 0.1s]
对于python 3.x与python2.x中新型类的继承特性总结的更多相关文章
- 关于Python中的类普通继承与super函数继承
关于Python中的类普通继承与super函数继承 1.super只能用于新式类 2.多重继承super可以保公共父类仅被执行一次 一.首先看下普通继承的写法 二.再看看super继承的写法 参考链接 ...
- python中的类与继承
Class 类的定义以及实例的建立 Python中,类通过 class 关键字定义. 例如最简单的一个类定义可以为: class Person(object): pass Python 的编程习惯,类 ...
- python 全栈开发,Day117(popup,Model类的继承,crm业务开发)
昨日内容回顾 第一部分:权限相关 1. 权限基本流程 用户登录成功后获取权限信息,将[权限和菜单]信息写入到session. 以后用户在来访问,在中间件中进行权限校验. 为了提升用户体验友好度,在后台 ...
- JavaScript中的类式继承和原型式继承
最近在看<JavaScript设计模式>这本书,虽然内容比较晦涩,但是细品才发现此书内容的强大.刚看完第四章--继承,来做下笔记. 书中介绍了三种继承方式,类式继承.原型式继承和掺元类继承 ...
- java中阻止类的继承
1.使用final来修饰类 final表示这个类是继承树的末端,不能被继承. 2.将类的构造方法声明为private的,再提供一个static的方法来返回一个类的对象. JAVA语言要求继承时必须在构 ...
- ES6中的类和继承
class的写法及继承 JavaScript 语言中,生成实例对象的传统方法是通过构造函数.下面是一个例子 function Point(x, y) { this.x = x; this. ...
- [Android Studio] Android Studio中查看类的继承关系
转载自:http://blog.csdn.net/hyr83960944/article/details/38098091 查看类的继承关系的快捷键F4,在Android Studio常用快捷键这篇文 ...
- Android Studio中查看类的继承关系
查看类的继承关系的快捷键F4.在Android Studio经常使用快捷键这篇文章中.有写了.今天主要是讲一些关于这个快捷键出来的界面的一些配置.这块功能相对偏冷一些,可能非常多人都会用不到.可是关于 ...
- ES5中的类与继承
最近在重新复习TypeScript,看到类这块的时候自然会和ES5中的类写法进行对比加深印象. 发现ES5的类与继承一些细节还是挺多的,时间久了容易忘记,特此记录下. 首先是ES5的类定义,这没什么好 ...
随机推荐
- scrapy 中没有 crawl 命令
确保两点: 1.把爬虫.py 复制到 spider 文件夹里 如 执行 scrapy crawl demo.py (spiders 中就要有 demo.py 文件) 2.在项目文件夹内执行命令 在 s ...
- 吴裕雄 Bootstrap 前端框架开发——Bootstrap 表单:内联表单
<!DOCTYPE html> <html> <head> <meta charset="utf-8"> <title> ...
- C#文件过滤器filter---转载
C#文件过滤器filter OpenFileDialog对话框的Filter属性说明: 首先说明一个示例,分析一下Filter属性的构成:“ Excel文件|*.xls ”,前面的“Excel文件”成 ...
- SQL注入个人理解及思路(包括payload和绕过的一些方式)
首先本文主要是把我对SQL注入的一些坑和最早学习SQL注入的时候的一些不理解的地方做一个梳理. (本文仅为个人的一点皮毛理解,如有错误还望指出,转载请说明出处,大佬勿喷=.=) 什么是SQL注入呢? ...
- Django:使用django自带的登录模块登录后会默认登录到 /accounts/profile 下的问题
django settings中LOGIN_REDIRECT_URL默认重定向到/accounts/profile下,可通过配置修改
- VUE.JS和小程序的共通之处
我是先学习的小程序开发,之后才了解到vue.js.也是一直没有时间去看相关vue.js的知识和内容.现在回顾起来,小程序和vue.js都是前端的内容. 例如小程序中的目录结构物page+app.js+ ...
- web页面的JS部分乱码了!!
在引用的地方给出编码即可 <script type='text/javascript' src='/zbjc/resources/normal/index_page.js' charset=&q ...
- Golang基础之文件操作
目录 文件读取 os.open文件读取 os.open循环读取 bufio文件读取 bufio循环读取 ioutil读取整个文件 文件写入 os.OpenFile文件写入 bufio.NewWrite ...
- [易语言][ExDui][Tutorial]1.NameSelector
咕咕咕 尝试自己写组件对象被易语言的对象劝退后,我又回来写教程了. 相信上一章对如何创建窗口讲得足够透彻了,这一章上项目实战:点名器. 点名器这种简单的东西实在是经常被拿出来开刀啊. 还有一点,发现之 ...
- Day9 - A - Apple Catching POJ - 2385
Description 有两棵APP树,编号为1,2.每一秒,这两棵APP树中的其中一棵会掉一个APP.每一秒,你可以选择在当前APP树下接APP,或者迅速移动到另外一棵APP树下接APP(移动时间可 ...