_new__和__init__的区别

__new__是Python面向对象语言中一个很少用的函数,更多使用的是__init__这个函数。例如:

class Book(object):
    def __init__(self, title):
        super(Book, self).__init__(self)
        self.title = title

# Define a book

b = Book(‘The Django Book‘)
print b.title
 

上面算是OOP语言的入门代码了,粗略一看__init__和java中的构造函数一样,其实不然,实际上它根本不能算的上构造函数。__new__才是创建实例的方法。

根据官方文档:

  • __init__是当实例对象创建完成后被调用的,然后设置对象属性的一些初始值。
  • __new__是在实例创建之前被调用的,因为它的任务就是创建实例然后返回该实例,是个静态方法。

也就是,__new__在__init__之前被调用,__new__的返回值(实例)将传递给__init__方法的第一个参数,然后__init__给这个实例设置一些参数。

class Book(object):
    def __new__(cls, title):
        print ‘__new__‘
        return super(Book, cls).__new__(cls)

    def __init__(self, title):
        print ‘__init__‘
        super(Book, self).__init__(self)
        self.title = title

b = Book(‘The Django Book‘)
print b.title
 

上面执行的结果:

__new__
__init__
The Django Book
 

__new__的应用场景

官方文档指出__new__方法的两种用法。

允许继承不可变类型(str,int, tuple)

关于这种也有比较多的例子,网上搜到的例子基本上都属于理论性,实际中用法不太常见。

在MetaClass中使用

MetaClass算是Python的语法糖吧,简单来说通过它可以动态生成或更改class的定义。

一个比较实际的例子,是在Django admin 表单验证的时候如何访问当前请求request。StackFlow的链接如下:

http://stackoverflow.com/questions/1057252/how-do-i-access-the-request-object-or-any-other-variable-in-a-forms-clean-met/6062628#6062628

首先想到的是把request也传递过去,在clean方法就可以使用了。

class MyForm(forms.ModelForm):
    def __init__(self, *args, **kwargs):
        self.request = kwargs.pop(‘request‘)
        super(MyForm, self).__init__(*args, **kwargs)

    def clean(self):
        #这里可以得到self.request的信息
        pass
 

在平常的view用下面的代码调用:

f = MyForm(request.POST, request=request)
 

但是在定制ModelAdmin的时候却不行,因为admin只提供get_form这个方法,返回值是类对象,而不是实例对象

get_form(self, request, *args, **kwargs):
    # 这行代码是错误的
    # return MyForm(request=request)
    return MyForm     # OK
 

用__new__方法可以解决这个问题。

def get_form(self, request, *args, **kwargs):
    class ModelFormMetaClass(MyForm):
        def __new__(cls, *args, **kwargs):
            kwargs[‘request‘] = request
            return MyForm(*args, **kwargs)
    return ModelFormMetaClass
 

那么结果如何呢,add_view的调用代码如下:

def add_view(self, request, form_url=‘‘, extra_context=None)"
    ...
    ModelForm = self.get_form(request)
    if request.method == ‘POST‘:
        form = ModelForm(request.POST, request.FILES)
        #可以获取request参数
        # print form.request
        if form.is_valid():
            pass
        else:
            pass
    else:
        ...(计算initial)
        form = ModelForm(initial=initial)
 

分析:form = ModelFormMetaClass(request.POST, request.FILES),按照通常的理解右边应该返回的是ModelFormMetaClass的一个实例,由于重写了__new__函数,没有调用父类函数,而是直接返回了一个带有request参数的MyForm实例,然后调用__init__函数,因此最后ModelFormMetaClass()返回也是这个实例,而左边也需要的是MyForm的实例对象。因此__new__函数的作用是创建一个实例。

备注:MetaClass它会降低代码的可读性,也有替代方案,不建议项目中使用。有兴趣的话可以参考这里

原文地址:https://www.cnblogs.com/chaojiyingxiong/p/9819264.html

时间: 2024-10-05 05:23:51

_new__和__init__的区别的相关文章

python类中super()和__init__()的区别

本文和大家分享的主要是python开发中super()和__init__()的区别,希望通过本文的分享,对初学者学习这部分内容有所帮助. 1.单继承时super()和__init__()实现的功能是类似的 class Base(object): def __init__(self): print 'Base create' class childA(Base): def __init__(self): print 'creat A ', Base.__init__(self) class chi

python中super().__init__和类名.__init__的区别

super().__init__相对于类名.__init__,在单继承上用法基本无差 但在多继承上有区别,super方法能保证每个父类的方法只会执行一次,而使用类名的方法会导致方法被执行多次 多继承时,使用super方法,对父类的传参数,应该是由于python中super的算法导致的原因,必须把参数全部传递,否则会报错 单继承时,使用super方法,则不能全部传递,只能传父类方法所需的参数,否则会报错 多继承时,相对于使用类名.__init__方法,要把每个父类全部写一遍, 而使用super方法

python__new__与__init__的区别

__new__ __init__区别 1 class A(object): 2 def __init__(self,*args, **kwargs): 3 print "init A" 4 def __new__(cls,*args, **kwargs): 5 print "new A %s"%cls 6 #return super(A, cls).__new__(cls, *args, **kwargs) 7 return object.__new__(cls,

__init__ __new__区别

请运行代码: class A: def __init__(self): print "A.__init" def __new__(self): print "A.__new" class B(object): def __init__(self): print "B.__init" super(B, self).__init__() def __new__(cls): print "B.__new__" return supe

__new__和__init__的区别

__new__是一个静态方法,而__init__是一个实例方法. __new__方法会返回一个创建的实例,而__init__什么都不返回. 只有在__new__返回一个cls的实例时后面的__init__才能被调用,并且__init__方法的属性是__new__返回的实例 4.当创建一个新实例时调用__new__,初始化一个实例时用__init__. 单例类 class Singleton(object): def __new__(cls, *args, **kwargs): # 如果没有_in

__new__与__init__的区别

__new__  : 控制对象的实例化过程 , 在__init__方法之前调用 __init__ : 对象实例化对象进行属性设置 class User: def __new__(cls, *args, **kwargs): """控制对象的创建的过程,在__init__方法之前调用""" print('invoke __new__') return super().__new__(cls) # 需要返回cls, 不然是不会调用__init__方法

python __new__, __init__,__call__区别

转载link:http://pycoders-weekly-chinese.readthedocs.io/en/latest/issue6/a-guide-to-pythons-magic-methods.htmlc

python中的__new__与__init__,新式类和经典类(2.x)

在python2.x中,从object继承得来的类称为新式类(如class A(object))不从object继承得来的类称为经典类(如class A()) 新式类跟经典类的差别主要是以下几点: 1. 新式类对象可以直接通过__class__属性获取自身类型:type 2. 继承搜索的顺序发生了改变,经典类多继承时属性搜索顺序: 先深入继承树左侧,再返回,开始找右侧(即深度优先搜索);新式类多继承属性搜索顺序: 先水平搜索,然后再向上移动 例子: 经典类: 搜索顺序是(D,B,A,C)>>&

深入理解Python中的 __new__ 和 __init__

本文为译文,原文链接:https://spyhce.com/blog/understanding-new-and-init 本文的目的是讨论Python中 __new__ 和 __init___ 的用法. __new__ 和 __init__ 的区别主要表现在:1. 它自身的区别:2. 及在Python中新式类和老式类的定义. 理解 __new__ 和 __init__ 的区别 这两个方法的主要区别在于:__new__ 负责对象的创建而 __init__ 负责对象的初始化.在对象的实例化过程中,