Python之经典类VS新式类和Supper

news/2024/10/23 12:25:46/
  • 问题:

把下面代码用python2 和python3都执行一下

   
  1. #_*_coding:utf-8_*_
  2. class A:
  3. def __init__(self):
  4. self.n = 'A'
  5. class B(A):
  6. # def __init__(self):
  7. # self.n = 'B'
  8. pass
  9. class C(A):
  10. def __init__(self):
  11. self.n = 'C'
  12. class D(B,C):
  13. # def __init__(self):
  14. # self.n = 'D'
  15. pass
  16. obj = D()
  17. print(obj.n)
输出结果
    
  1. D:\Python\python\python-2.7.13\Python27\python2.exe
  2. A
    
  1. D:\Python\python\python-3.6.1\Python36-64\python.exe
  2. C

  • 原因:
     
  1. classical vs new style
  2. 经典类:深度优先
  3. 新式类:广度优先
  4.          
    1. 1)首先,他们写法不一样:
    2. class A:
    3. pass
    4. class B(object):
    5. pass
    6. 2)在多继承中,新式类采用广度优先搜索,而旧式类是采用深度优先搜索。
    7. 3)新式类更符合OOP编程思想,统一了python中的类型机制。


  • 分析:

从Python2.2开始,Python 引入了 new style class(新式类)

    

新式类跟经典类的差别主要是以下几点:

                                  
  1. 新式类对象可以直接通过__class__属性获取自身类型:type
  2. 继承搜索的顺序发生了改变,经典类多继承属性搜索顺序: 先深入继承树左侧,再返回,开始找右侧;新式类多继承属性搜索顺序: 
  3. 先水平搜索,然后再向上移动;
  4. 新式类增加了__slots__内置属性, 可以把实例属性的种类锁定到__slots__规定的范围之中;
  5. 新式类增加了__getattribute__方法;


  • 举例说明
    • 新式类对象可以直接通过__class__属性获取自身类型:type;

      
  1. # -*- coding:utf-8 -*-
  2. class E:
  3. # 经典类
  4. pass
  5. class E1(object):
  6. # 新式类
  7. pass
  8. e = E()
  9. print("经典类")
  10. print(e)
  11. print(type(e))
  12. print(e.__class__)
  13. print("新式类")
  14. e1 = E1()
  15. print(e1)
  16. print(type(e1))
  17. print(e1.__class__)

输出结果:

  • pyhon2.7
       
  1. 经典类
  2. <__main__.E instance at 0x0000000002ABF148>
  3. <type 'instance'>
  4. __main__.E
  5. 新式类
  6. <__main__.E1 object at 0x0000000002AB65C0>
  7. <class '__main__.E1'>
  8. <class '__main__.E1'>

  • pyhon3.6

       
  1. 经典类
  2. <__main__.E object at 0x000000000267B7F0>
  3. <class '__main__.E'>
  4. <class '__main__.E'>
  5. 新式类
  6. <__main__.E1 object at 0x000000000267B3C8>
  7. <class '__main__.E1'>
  8. <class '__main__.E1'>

  • Python 2.x中默认都是经典类,只有显式继承了object才是新式类

  • Python 3.x中默认都是新式类,不必显式的继承object

  • E1是定义的新式类。那么输输出e1的时候,不论是type(e1),还是e1.__class__都是输出的<class '__main__.E1'>。



    • 继承搜索的顺序发生了改变,经典类多继承属性搜索顺序:先深入继承树左侧,再返回,开始找右侧;新式类多继承属性搜索顺序: 先水平搜索,然后再向上移动;
      
  1. # -*- coding:utf-8 -*-
  2. class A(object):
  3. """
  4. 新式类
  5. 作为所有类的基类
  6. """
  7. def foo(self):
  8. print("class A")
  9. class A1():
  10. """
  11. 经典类
  12. 作为所有类的基类
  13. """
  14. def foo(self):
  15. print("class A1")
  16. class C(A):
  17. pass
  18. class C1(A1):
  19. pass
  20. class D(A):
  21. def foo(self):
  22. print("class D")
  23. class D1(A1):
  24. def foo(self):
  25. print("class D1")
  26. class E(C, D):
  27. pass
  28. class E1(C1, D1):
  29. pass
  30. e = E()
  31. e.foo()
  32. e1 = E1()
  33. e1.foo()

  • pyhon2.7
       
  1. class D
  2. class A1

  • pyhon3.6
       
  1. class D
  2. class D1

因为A新式类,对于继承A类都是新式类,首先要查找类E中是否有foo(),如果没有则按顺序查找C->D->A。它是一种广度优先查找方式。
因为A1经典类,对于继承A1类都是经典类,首先要查找类E1中是否有foo(),如果没有则按顺序查找C1->A1->D1。它是一种深度优先查找方式。

    • 新式类增加了__slots__内置属性, 可以把实例属性的种类锁定到__slots__规定的范围之中;
                    比如只允许对A实例添加name和age属性:

      
  1. # -*- coding:utf-8 -*-
  2. class A(object):
  3. __slots__ = ('name', 'age')
  4. class A1():
  5. __slots__ = ('name', 'age')
  6. a1 = A1()
  7. a = A()
  8. a1.name1 = "a1"
  9. a.name1 = "a"
  • pyhon2.7
        
  1. Traceback (most recent call last):
  2. File "test.py", line 12, in <module>
  3. a.name1 = "a"
  4. AttributeError: 'A' object has no attribute 'name1'


A是新式类添加了__slots__ 属性,所以只允许添加 name age

A1经典类__slots__ 属性没用,

所以a.name是会出错的


  • pyhon3.6
    
  1. Traceback (most recent call last):
  2. File "test.py", line 11, in <module>
  3. a1.name1 = "a1"
  4. AttributeError: 'A1' object has no attribute 'name1'

在python3.x中,

A和

A1都是新式类

    
  1. 通常每一个实例都会有一个__dict__属性,用来记录实例中所有的属性和方法,也是通过这个字典,可以让实例绑定任意的属性
  2. __slots__属性作用就是,当类C有比较少的变量,而且拥有__slots__属性时,
  3. C的实例 就没有__dict__属性,而是把变量的值存在一个固定的地方。如果试图访问一个__slots__中没有
  4. 的属性,实例就会报错。
  5. 这样操作有什么好处呢?__slots__属性虽然令实例失去了绑定任意属性的便利,
  6. 但是因为每一个实例没有__dict__属性,却能有效节省每一个实例的内存消耗,有利于生成小而精
  7. 干的实例。


    • 新式类增加了__getattribute__方法;
     
  1. class A(object):
  2. def __getattribute__(self, *args, **kwargs):
  3. print("A.__getattribute__")
  4. class A1():
  5. def __getattribute__(self, *args, **kwargs):
  6. print("A1.__getattribute__")
  7. a1 = A1()
  8. a = A()
  9. a.test
  10. print("=========")
  11. a1.test

  • pyhon2.7
       
  1. A.__getattribute__
  2. Traceback (most recent call last):
  3. =========
  4. File "test.py", line 15, in <module>
  5. a1.test
  6. AttributeError: A1 instance has no attribute 'test'

  • pyhon3.6
       
  1. A.__getattribute__
  2. =========
  3. A1.__getattribute__

可以看出A是新式类,每次通过实例访问属性,都会经过__getattribute__函数,

A1不会调用__getattribute__所以出错了


总之

    
  1. Python 2.x中默认都是经典类,只有显式继承了object才是新式类
  2. Python 3.x中默认都是新式类,不必显式的继承object

super()


     
  1. super(self,C).func() #调用的并不是其父类C的func,而是C在MRO中的下一个类的func,
  2. 不能再经典类中使用
  3. 开始一直以为在多重继承的情况下选择执行某个父类的方法,网上有不少也是这么说的(被误导了)。
  4. 经典类的MRO(基类搜索顺序)算法是深度优先。

    新式类的MRO算法是C3算法。


     
  1. class A(object):
  2. def __init__(self):
  3. print "A"
  4. class B(object):
  5. def __init__(self):
  6. print ("B")
  7. class C(object):
  8. def __init__(self):
  9. print ("C")
  10. class D(A, B, C):
  11. def __init__(self):
  12. super(D, self).__init__()
  13. super(A, self).__init__()
  14. super(B, self).__init__()
  15. super(C, self).__init__()
  16. X = D()

输出结果:
     
  1. A
  2. B
  3. C

会发现:

      
  1. super(D,self).__init__()

执行的是A.__init__()


      
  1. super(A,self).__init__()
执行的是B.__init__()

      
  1. super(B,self).__init__()

执行的是C.__init__()


      
  1. super(C,self).__init__()
执行的是Object.__init__()

这是因为mro(D)为:[ D, A, B, C, Object]



http://www.ppmy.cn/news/508053.html

相关文章

ESP32-S2 Linux串口OTA升级

一、简介 OTA(Over-the-Air Technology )空中下载技术&#xff0c;通过远程网络为设备更新或者升级&#xff0c;本质就是把固件通过WiFi、蓝牙、4G、5G、网线等写到MCU flash里面&#xff0c;最终实现升级。 二、原理 1.终端设备发送需要升级的固件&#xff1b; 2.MCU接收来自…

65C02指令说明书

65C02 &#xff08;Pinokio163.com 译、编 2002-6-17&#xff09; 概要 指令集 寻址模式 指令编码 概要 65C02有16位&#xff08;64K&#xff09;的地址空间分成256个页面&#xff0c;每个256字节。0 页面($0000-$00FF)有一些很特别的性质&#xff0c;例如寻址模式&#xff0…

Tensorflow图像识别-2

文章目录 创建tensorflow虚拟环境测试 安装TensorFlow Object Detection API(对象检测API)目录安排 编译protoc测试一下结果如图 tensorflow图像识别-1 labelimg Tensorflow图像识别-2  试一下识别 Tensorflow图像识别-3  训练&#xff0c;笔记本要退休的感觉 Tensorflow图像…

pix2pix_tensorflow下,代码解读

pix2pix.py_tensorflow代码解读 pix2pix.py为&#xff1a; from __future__ import absolute_import from __future__ import division from __future__ import print_functionimport tensorflow as tf import numpy as np import argparse import os import json import glo…

javaWeb-2 JSP

JSP JSP的概述 JSP&#xff08;全称JavaServer Pages&#xff09;是由Sun Microsystems公司主导创建的一种动态网页技术标准。JSP部署于网络服务器上&#xff0c;可以响应客户端发送的请求&#xff0c;并根据请求内容动态地生成HTML、XML或其他格式文档的Web网页&#xff0c;…

Android5.0 Gallery2上编译Gallery模块出错

在L上面&#xff0c;编译整个project可以编译通过&#xff0c;但是单独编译Gallery2模块出错&#xff0c;build gallery模块出现refocus的error target C: libjni_jpegstream < packages/apps/Gallery2/jni_jpegstream/src/outputstream_wrapper.cpp target C: libjni_jpe…

第 2 章 JUC

第 2 章 JUC 1、volatile 关键字 谈谈你对volatile的理解 1.1、volatile 三大特性 volatile是java虚拟机提供的轻量级同步机制 可以将 volatile 看作是乞丐版的 synchronized 锁 保证内存可见性禁止指令重排不保证原子性 1.2、JMM 内存模型 1.2.1、谈谈 JMM 谈谈 JMM JMM&a…

2.HTMLCSS

B站狂神视频总结 HTML、CSS、JavaScript HTML 1、初识HTML HyperTextMarkupLanguage&#xff08;超文本标记语言&#xff09;< body >、< /body>等成对的标签&#xff0c;分别叫做开放标签和闭合标签&#xff0c;单独呈现的标签&#xff08;空元素&#xff09;…