建立类的时候,需要使用_init_魔法方法,感到好奇,查阅资料后,总结一下。
_new_(cls[,...])
当对象创建时,第一个调用的方法就是_new_,而前面说到的_init_方法更像是一个配置文件。_new_才是真正的构造函数,创建并返回一个实例对象,如果_new_只调用了一次,就会得到一个对象。它的第一个参数是这个类,其他的参数是用来直接传递给_init_ 方法。
- _new_ 是在一个对象实例化的时候所调用的第一个方法
- 它的第一个参数是这个类,其他的参数是用来直接传递给 _init_ 方法
- _new_ 决定是否要使用该 _init_ 方法,因为 _new_ 可以调用其他类的构造方法或者直接返回别的实例对象来作为本类的实例,如果 _new_ 没有返回实例对象,则 _init_ 不会被调用
- _new_ 主要是用于继承一个不可变的类型比如一个 tuple 或者 string
class TestClass:
def __init__(self):
print("调用__init__")
def __new__(cls, *args, **kwargs):
print("调用__new__")
return object.__new__(cls)#返回当前类的实例
t = TestClass()
上下文管理器
可能有人好奇上下文管理器是什么,和魔法方法有啥关系?平时读取文件的时候,常常说要使用with open()来进行文件内容的读取,而with后跟着的就是上下文管理器,简单的说,任何实现了_enter_方法和_exit_方法的对象都可以称之为上下文管理器。它可以在执行with 包裹的执行代码(也就是自己写的那部分)前后执行对应的_enter_方法或_exit_方法,可以想到的就是open()执行的_enter_是打开文件,而_exit_则是关闭了文件。顺便说一句,python还提供了一个contextmanager的装饰器,更进一步简化了上下文管理器的实现方式。
class TestClass:
def __init__(self,name):
self.name = name
def __enter__(self):
print("%s开始工作啦!"%self.name)
def __exit__(self, exc_type, exc_val, exc_tb):
if exc_type is None: # 可以利用exc_type来捕获异常
print("%s结束工作啦!"%self.name)
else:
print("工作出错了!")
print('Type: ', exc_type)
print('Value:', exc_val)
print('TreacBack:', exc_tb)
# 返回值决定了捕获的异常是否继续向外抛出
# 如果是 False 那么就会继续向外抛出,程序会看到系统提示的异常信息
# 如果是 True 不会向外抛出,程序看不到系统提示信息,只能看到else中的输出
return True
t = TestClass(name = "卡比兽")
with t :
raise NameError
print:
卡比兽开始工作啦!
工作出错了!
('Type: ', <type 'exceptions.NameError'>)
('Value:', NameError())
('TreacBack:', <traceback object at 0x0000000003CF9348>)
待更新。。。
参考链接