Python面试之理解__new__和__init__的区别

很多同学都以为Python中的 __init__ 是构造方法,但其实不然,Python中真正的构造方法是__new____init____new__ 有什么区别?本文就来探讨一下。

我们先来看一下 __init__ 的用法

class Person(object):
    def __init__(self, name, age):
        print("in __init__")
        self._name = name
        self._age = age 

p = Person("Wang", 33) 

上面的代码会输出如下的结果

in __init__
<__main__.Person object at 0x7fb2e0936450>

那么我们思考一个问题,Python中要实现Singleton怎么实现,要实现工厂模式怎么实现?

__init__ 函数似乎没法做到呢~

实际上, __init__ 函数并不是真正意义上的构造函数, __init__ 方法做的事情是在对象创建好之后初始化变量。真正创建实例的是 __new__ 方法。

我们来看下面的例子

class Person(object):
    def __new__(cls, *args, **kwargs):
        print("in __new__")
        instance = object.__new__(cls, *args, **kwargs)
        return instance

    def __init__(self, name, age):
        print("in __init__")
        self._name = name
        self._age = age

p = Person("Wang", 33)

上面的代码输出如下的结果

in __new__
in __init__

上面的代码中实例化了一个Person对象,可以看到 __new____init__ 都被调用了。 __new__ 方法用于创建对象并返回对象,当返回对象时会自动调用 __init__ 方法进行初始化。 __new__ 方法是静态方法,而 __init__ 是实例方法。

好了,理解 __new____init__ 的区别后,我们再来看一下前面提出的问题,用Python怎么实现Singleton,怎么实现工厂模式?

先来看Singleton

class Singleton(object):
    _instance = None
    def __new__(cls, *args, **kwargs):
        if cls._instance is None:
            cls._instance = object.__new__(cls, *args, **kwargs)

        return cls._instance

s1 = Singleton()
s2 = Singleton()
print(s1)
print(s2) 

上面的代码输出

<__main__.Singleton object at 0x7fdef58b1190>
<__main__.Singleton object at 0x7fdef58b1190>

可以看到s1和s2都指向同一个对象,实现了单例模式。

再来看下工厂模式的实现

class Fruit(object):
    def __init__(self):
        pass

    def print_color(self):
        pass

class Apple(Fruit):
    def __init__(self):
        pass

    def print_color(self):
        print("apple is in red")

class Orange(Fruit):
    def __init__(self):
        pass

    def print_color(self):
        print("orange is in orange")

class FruitFactory(object):
    fruits = {"apple": Apple, "orange": Orange}

    def __new__(cls, name):
        if name in cls.fruits.keys():
            return cls.fruits[name]()
        else:
            return Fruit()

fruit1 = FruitFactory("apple")
fruit2 = FruitFactory("orange")
fruit1.print_color()    
fruit2.print_color()    

上面的代码输出

apple is in red
orange is in orange

看完上面两个例子,大家是不是对 __new____init__ 的区别有了更深入的理解?