澳门新萄京官方网站-www.8455.com-澳门新萄京赌场网址

python自带的三个装饰器,PYTHON中__NEW__和__INIT__的

2019-05-02 作者:www.8455.com   |   浏览(61)

__init__与__new__区别:

python中的__init__ 、__new__、__call__小结,__init___new__

这篇文章主要介绍了python中的__init__ 、__new__、__call__小结,需要的朋友可以参考下

1.__new__(cls, *args, **kwargs)  创建对象时调用,返回当前对象的一个实例;注意:这里的第一个参数是cls即class本身
2.__init__(self, *args, **kwargs) 创建完对象后调用,对当前对象的实例的一些初始化,无返回值,即在调用__new__之后,根据返回的实例初始化;注意,这里的第一个参数是self即对象本身【注意和new的区别】
3.__call__(self,  *args, **kwargs) 如果类实现了这个方法,相当于把这个类型的对象当作函数来使用,相当于 重载了括号运算符  

看具体的例子:

复制代码 代码如下:
class O(object):

    def __init__(self, *args, **kwargs):
        print "init"
        super(O, self).__init__(*args, **kwargs)

    def __new__(cls, *args, **kwargs):
        print "new", cls
        return super(O, cls).__new__(cls, *args, **kwargs)

    def __call__(self,  *args, **kwargs):
        print "call"       

    oo = O()
    print "________"
    oo() 

打印出来的是:
复制代码 代码如下:
new
init
________
call
比如:Python Singleton(单例模式)实现,那我们是不是只是重载一些__new__方法就可以了
复制代码 代码如下:
class Singleton1(object):
    """ 重载new方法"""
    def __new__(cls, *args, **kwargs):
        if not "_instance" in vars(cls):
            cls._instance = super(Singleton1, cls).__new__(cls, *args, **kwargs)
        return cls._instance

可不可以重载__init__方法呢?明显不可以,因为__init__之前调用了__new__方法,这时候已经生成了一个对象了,没办法实现单例模式

===========================================  

注意1、__init__并不相当于C#中的构造函数,执行它的时候,实例已构造出来了。

1 2 3 4 5 class A(object):     def __init__(self,name):         self.name=name     def getName(self):         return 'A ' self.name

当我们执行

1 a=A('hello')

时,可以理解为

1 2 a=object.__new__(A) A.__init__(a,'hello')

即__init__作用是初始化已实例化后的对象。

注意2、子类可以不重写__init__,实例化子类时,会自动调用超类中已定义的__init__

1 2 3 4 5 6 7 class B(A):     def getName(self):         return 'B ' self.name   if __name__=='__main__':     b=B('hello')     print b.getName()

但如果重写了__init__,实例化子类时,则不会隐式的再去调用超类中已定义的__init__

1 2 3 4 5 6 7 8 9 class C(A):     def __init__(self):         pass     def getName(self):         return 'C ' self.name   if __name__=='__main__':     c=C()     print c.getName()

则会报"AttributeError: 'C' object has no attribute 'name'”错误,所以如果重写了__init__,为了能使用或扩展超类中的行为,最好显式的调用超类的__init__方法

1 2 3 4 5 6 7 8 9 class C(A):     def __init__(self,name):         super(C,self).__init__(name)     def getName(self):         return 'C ' self.name   if __name__=='__main__':     c=C('hello')        print c.getName()    

、__new__、__call__小结,__init___new__ 这篇文章主要介绍了python中的__init__ 、__new__、__call__小结,需要的朋友可以参考下 1.__new...

说到装饰器,就不得不说python自带的三个装饰器:

这篇文章主要介绍了python中的__init__ 、__new__、__call__小结,需要的朋友可以参考下

__NEW__和__INIT__的区别例子

__init__在python,其实是,在实例化之后执行的,用来初始化一些属性,相当于构造函数,但是又不一样

1、@property   将某函数,做为属性使用

1.__new__(cls, *args, **kwargs)  创建对象时调用,返回当前对象的一个实例;注意:这里的第一个参数是cls即class本身
2.__init__(self, *args, **kwargs) 创建完对象后调用,对当前对象的实例的一些初始化,无返回值,即在调用__new__之后,根据返回的实例初始化;注意,这里的第一个参数是self即对象本身【注意和new的区别】
3.__call__(self,  *args, **kwargs) 如果类实现了这个方法,相当于把这个类型的对象当作函数来使用,相当于 重载了括号运算符  

Python中__new__和__init__用得最多的是__init__,__init__很像PHP,C,JAVA等其他语言中的构造方法,但是__new__也类似于构造方法,而且比__init__更早执行,它们有什么区别?怎么用?

细心一些,通过参数会有所发现,其实__init__(self)  self隐式的将,实例传过来。

 @property 修饰,就是将方法,变成一个属性来使用。

看具体的例子:

__init__ 方法是什么?

 

class A():


    @property
    def pfunc(self):
        return self.value

    @pfunc.setter
    def pfunc(self,value):
        self.value = value

    @property
    def pfunc1(self):
        print('this is property')

if __name__=="__main__":

    A.pfunc = 9
    print A.pfunc
    A.pfunc1

复制代码 代码如下:

使用Python写过面向对象的代码的同学,可能对 __init__ 方法已经非常熟悉了,__init__ 方法通常用在初始化一个类实例的时候。例如:

__new__在python中其实是,在实例化之前执行的,这个通过参数一样可以看出

 

class O(object):

# -*- coding: utf-8 -*-

__new__(cls),cls是隐式的传递的类对象,并不是实例。因为__new__的任务就是,创建类实例并返回实例。

2、@classmethod  修饰类的方式

    def __init__(self, *args, **kwargs):
        print "init"
        super(O, self).__init__(*args, **kwargs)

class Person(object):
    """Silly Person"""

class temp(object):

    def __init__(self,txt):
        self.txt = txt
        print '__init__'


    def __new__(cls,txt):
        print '__new__'
        print txt
        return super(temp,cls).__new__(cls)

temp('what?')

带修饰类方法:cls做为方法的第一个参数,隐式的将类做为对象,传递给方法,调用时无须实例化。

    def __new__(cls, *args, **kwargs):
        print "new", cls
        return super(O, cls).__new__(cls, *args, **kwargs)

    def __python自带的三个装饰器,PYTHON中__NEW__和__INIT__的区别例子。init__(self, name, age):
        self.name = name
        self.age = age

结果:

普通函数方法:self做为第一个参数,隐式的将类实例传递给方法,调用方法时,类必须实例化。

    def __call__(self,  *args, **kwargs):
        print "call"       

    def __str__(self):
        return '<Person: %s(%s)>' % (self.name, self.age)

C:Python27python.exe D:/weixin/temp/abc.py
__new__
what?
__init__

Process finished with exit code 0
class A():
    def func(self,x,y):
        return x * y

    @classmethod
    def cfunc(cls,x,y):
        return x * y

if __name__=="__main__":
    print A().func(5,5)
    print A.cfunc(4,5)

    oo = O()
    print "________"
    oo() 

if __name__ == '__main__':
    piglei = Person('piglei', 24)
    print piglei
这样便是__init__最普通的用法了。但__init__其实不是实例化一个类的时候第一个被调用 的方法。当使用 Persion(name, age) 这样的表达式来实例化一个类时,最先被调用的方法 其实是 __new__ 方法。

 

 

打印出来的是:

__new__ 方法是什么?
__new__方法接受的参数虽然也是和__init__一样,但__init__是在类实例创建之后调用,而 __python自带的三个装饰器,PYTHON中__NEW__和__INIT__的区别例子。new__方法正是创建这个类实例的方法。

 

3、@staticmethod  修饰类的方式

复制代码 代码如下:

# -*- coding: utf-8 -*-

Linux and python学习交流1,2群已满.

1)是把函数嵌入到类中的一种方式,函数就属于类,同时表明函数不需要访问这个类

new
init
________
call

class Person(object):
    """Silly Person"""

Linux and python学习交流3群新开,欢迎加入,一起学习.qq 3群:563227894

 2)使用修饰服,修饰方法,不需要实例化

比如:Python Singleton(单例模式)实现,那我们是不是只是重载一些__new__方法就可以了

    def __new__(cls, name, age):
        print '__new__ called.'
        return super(Person, cls).__new__(cls, name, age)

不前进,不倒退,停止的状态是没有的.

 

复制代码 代码如下:

    def __init__(self, name, age):
        print '__init__ called.'
        self.name = name
        self.age = age

一起进步,与君共勉,

class A():
    def func(self,x,y):
        return x * y


    @staticmethod
    def sfunc(x,y):
        return x * y


if __name__=="__main__":

    print A.sfunc(6,5)

class Singleton1(object):
    """ 重载new方法"""
    def __new__(cls, *args, **kwargs):
        if not "_instance" in vars(cls):
            cls._instance = super(Singleton1, cls).__new__(cls, *args, **kwargs)
        return cls._instance

    def __str__(self):
        return '<Person: %s(%s)>' % (self.name, self.age)

 

 

可不可以重载__init__方法呢?明显不可以,因为__init__之前调用了__new__方法,这时候已经生成了一个对象了,没办法实现单例模式

if __name__ == '__main__':
    piglei = Person('piglei', 24)
    print piglei
执行结果:

 

===========================================

piglei@macbook-pro:blog$ python new_and_init.py
__new__ called.
__init__ called.
<Person: piglei(24)>
通过运行这段代码,我们可以看到,__new__方法的调用是发生在__init__之前的。其实当 你实例化一个类的时候,具体的执行逻辑是这样的:

Linux and python学习交流1,2群已满.

 

p = Person(name, age)
首先执行使用name和age参数来执行Person类的__new__方法,这个__new__方法会 返回Person类的一个实例(通常情况下是使用 super(Persion, cls).__new__(cls, ... ...) 这样的方式),
然后利用这个实例来调用类的__init__方法,上一步里面__new__产生的实例也就是 __init__里面的的 self
所以,__init__ 和 __new__ 最主要的区别在于:

Linux and python学习交流3群新开,欢迎加入,一起学习.qq 3群:563227894

注意1、__init__并不相当于C#中的构造函数,执行它的时候,实例已构造出来了。

__init__ 通常用于初始化一个新实例,控制这个初始化的过程,比如添加一些属性, 做一些额外的操作,发生在类实例被创建完以后。它是实例级别的方法。
__new__ 通常用于控制生成一个新实例的过程。它是类级别的方法。
但是说了这么多,__new__最通常的用法是什么呢,我们什么时候需要__new__?

不前进,不倒退,停止的状态是没有的.

1
2
3
4
5
class A(object):
    def __init__(self,name):
        self.name=name
    def getName(self):
        return 'A ' self.name

__new__ 的作用
依照Python官方文档的说法,__new__方法主要是当你继承一些不可变的class时(比如int, str, tuple), 提供给你一个自定义这些类的实例化过程的途径。还有就是实现自定义的metaclass。

一起进步,与君共勉,

当我们执行

首先我们来看一下第一个功能,具体我们可以用int来作为一个例子:

 

1
a=A('hello')

假如我们需要一个永远都是正数的整数类型,通过集成int,我们可能会写出这样的代码。

时,可以理解为

class PositiveInteger(int):
    def __init__(self, value):
        super(PositiveInteger, self).__init__(self, abs(value))

1
2
a=object.__new__(A)
A.__init__(a,'hello')

i = PositiveInteger(-3)
print i
但运行后会发现,结果根本不是我们想的那样,我们任然得到了-3。这是因为对于int这种 不可变的对象,我们只有重载它的__new__方法才能起到自定义的作用。

即__init__作用是初始化已实例化后的对象。

这是修改后的代码:

注意2、子类可以不重写__init__,实例化子类时,会自动调用超类中已定义的__init__

class PositiveInteger(int):
    def __new__(cls, value):
        return super(PositiveInteger, cls).__new__(cls, abs(value))

1
2
3
4
5
6
7
class B(A):
    def getName(self):
        return 'B ' self.name
 
if __name__=='__main__':
    b=B('hello')
    print b.getName()

i = PositiveInteger(-3)
print i
通过重载__new__方法,我们实现了需要的功能。

但如果重写了__init__,实例化子类时,则不会隐式的再去调用超类中已定义的__init__

另外一个作用,关于自定义metaclass。其实我最早接触__澳门新萄京官方网站,new__的时候,就是因为需要自定义 metaclass,但鉴于篇幅原因,我们下次再来讲python中的metaclass和__new__的关系。

1
2
3
4
5
6
7
8
9
class C(A):
    def __init__(self):
        pass
    def getName(self):
        return 'C ' self.name
 
if __name__=='__main__':
    c=C()
    print c.getName()

用__new__来实现单例
事实上,当我们理解了__new__方法后,我们还可以利用它来做一些其他有趣的事情,比如实现 设计模式中的 单例模式(singleton) 。

则会报"AttributeError: 'C' object has no attribute 'name'”错误,所以如果重写了__init__,为了能使用或扩展超类中的行为,最好显式的调用超类的__init__方法

因为类每一次实例化后产生的过程都是通过__new__来控制的,所以通过重载__new__方法,我们 可以很简单的实现单例模式。

1
2
3
4
5
6
7
8
9
class C(A):
    def __init__(self,name):
        super(C,self).__init__(name)
    def getName(self):
        return 'C ' self.name
 
if __name__=='__main__':
    c=C('hello')   
    print c.getName()
 
 

class Singleton(object):
    def __new__(cls):
        # 关键在于这,每一次实例化的时候,我们都只会返回这同一个instance对象
        if not hasattr(cls, 'instance'):
            cls.instance = super(Singleton, cls).__new__(cls)
        return cls.instance

obj1 = Singleton()
obj2 = Singleton()

obj1.attr1 = 'value1'
print obj1.attr1, obj2.attr1
print obj1 is obj2
输出结果:

value1 value1
True
可以看到obj1和obj2是同一个实例。

本文由澳门新萄京官方网站发布于www.8455.com,转载请注明出处:python自带的三个装饰器,PYTHON中__NEW__和__INIT__的

关键词: