zoukankan      html  css  js  c++  java
  • Python新式类和经典类的区别

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

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

    1. 新式类对象可以直接通过__class__属性获取自身类型:type

    [python] view plain copy
     
    1. # -*- coding:utf-8 -*-    
    2.   
    3. class E:    
    4. #经典类  
    5.     pass  
    6.       
    7. class E1(object):    
    8. #新式类  
    9.     pass  
    10.        
    11. e = E()  
    12. print "经典类"  
    13. print e  
    14. print type(e)  
    15. print e.__class__  
    16.   
    17. print "新式类"  
    18. e1 = E1()  
    19. print e1  
    20. print e1.__class__  
    21. print type(e1)  
    [python] view plain copy
     
    1. 经典类  
    2. <__main__.E instance at 0x0000000002250B08>  
    3. <type 'instance'>  
    4. __main__.E  
    5.   
    6. 新式类  
    7. <__main__.E1 object at 0x0000000002248710>  
    8. <class '__main__.E1'>  
    9. <class '__main__.E1'>  

    我使用的是python 2.7。

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

    2. 继承搜索的顺序发生了改变,经典类多继承属性搜索顺序: 先深入继承树左侧,再返回,开始找右侧;新式类多继承属性搜索顺序: 先水平搜索,然后再向上移动

    [python] view plain copy
     
    1. # -*- coding:utf-8 -*-    
    2. class A(object):    
    3.     """ 
    4.     新式类 
    5.     作为所有类的基类 
    6.     """  
    7.     def foo(self):    
    8.         print "class A"   
    9.           
    10. class A1():    
    11.     """ 
    12.     经典类 
    13.     作为所有类的基类 
    14.     """  
    15.     def foo(self):    
    16.         print "class A1"    
    17.           
    18. class C(A):    
    19.     pass  
    20.       
    21. class C1(A1):    
    22.     pass  
    23.       
    24. class D(A):    
    25.     def foo(self):    
    26.         print "class D"    
    27.       
    28. class D1(A1):    
    29.     def foo(self):    
    30.         print "class D1"    
    31.           
    32.     
    33.     
    34. class E(C, D):    
    35.     pass  
    36.       
    37. class E1(C1, D1):    
    38.     pass  
    39.   
    40. e = E()  
    41. e.foo()  
    42.     
    43.   
    44. e1 = E1()  
    45. e1.foo()  

    输出

    [python] view plain copy
     
    1. class D  
    2. class A1  

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

    经典类

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

    经典类

    3. 新式类增加了__slots__内置属性, 可以把实例属性的种类锁定到__slots__规定的范围之中。

    比如只允许对A实例添加name和age属性:

    [python] view plain copy
     
    1. # -*- coding:utf-8 -*-    
    2.   
    3. class A(object):    
    4.     __slots__ = ('name', 'age')   
    5.   
    6. class A1():    
    7.     __slots__ = ('name', 'age')   
    8.       
    9. a1 = A1()  
    10. a = A()  
    11.   
    12. a1.name1 = "a1"  
    13. a.name1 = "a"  

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

    A1经典类__slots__ 属性没用,

    [python] view plain copy
     
    1. Traceback (most recent call last):  
    2.   File "t.py", line 13, in <module>  
    3.     a.name1 = "a"  
    4. AttributeError: 'A' object has no attribute 'name1'  

    所以a.name是会出错的

    通常每一个实例都会有一个__dict__属性,用来记录实例中所有的属性和方法,也是通过这个字典,可以让实例绑定任意的属性

    而__slots__属性作用就是,当类C有比较少的变量,而且拥有__slots__属性时,

    类C的实例 就没有__dict__属性,而是把变量的值存在一个固定的地方。如果试图访问一个__slots__中没有

    的属性,实例就会报错。这样操作有什么好处呢?__slots__属性虽然令实例失去了绑定任意属性的便利,

    但是因为每一个实例没有__dict__属性,却能有效节省每一个实例的内存消耗,有利于生成小而精

    干的实例。

    4. 新式类增加了__getattribute__方法

    [python] view plain copy
     
    1. class A(object):    
    2.     def __getattribute__(self, *args, **kwargs):    
    3.         print "A.__getattribute__"  
    4.           
    5.       
    6. class A1():    
    7.     def __getattribute__(self, *args, **kwargs):    
    8.         print "A1.__getattribute__"  
    9.           
    10.       
    11. a1 = A1()  
    12. a = A()  
    13.   
    14. a.test  
    15. print "========="  
    16. a1.test  
    [python] view plain copy
     
    1. A.__getattribute__  
    2. =========  
    3. Traceback (most recent call last):  
    4.   File "t.py", line 18, in <module>  
    5.     a1.test  
    6. AttributeError: A1 instance has no attribute 'test'  

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

    A1不会调用__getattribute__所以出错了

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

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

  • 相关阅读:
    PHP设计模式
    PHP设计模式
    PHP 23种设计模式
    MySQL 中的共享锁和排他锁的用法
    PHP_MySQL高并发加锁事务处理
    Connection: close和Connection: keep-alive有什么区别
    罗辑思维首席架构师:Go微服务改造实践
    真诚与尊重是技术团队的管理要点
    10种常见的软件架构模式
    百亿级微信红包的高并发资金交易系统设计方案
  • 原文地址:https://www.cnblogs.com/paranoia/p/6164361.html
Copyright © 2011-2022 走看看