with是从Python一个语法糖,它是一种上下文管理协议,目的在于把我们之前常见一个开发 try,except 和finally 关键字和一些文件开关闭合资源分配释放等问题都简化。

总结起来使用python 提供的with主要的作用是:

  • 实现自动调用对象资源的释放
  • 异常捕获和回滚
  • 减少用户手动调用的方法

常见的几个应用场景有:

  • 1、文件操作。
  • 2、进程线程之间互斥对象。
  • 3、支持上下文其他对象
  • 4、需要进行资源链接和结束的时候进行释放操作
  • 5、释放锁
  • 6、创建代码补丁

就是我们使用打开文件操作的时候 如果没有用with 就需要调用f.close关闭释放资源

With原理

本质上其实with是上下文管理:

  • 1、上下文管理协议 :
    • 包含方法enter() 和 exit(),支持该协议对象要实现这两个方法。
  • 2、上下文管理器 :
    • 定义执行with语句时要建立的运行时上下文,负责执行with语句块上下文中的进入与退出操作。
  • 3、进入上下:
    • 执行enter方法
  • 4、运行时发生了异常或结束是:
    • 退出上下文管理器,调用管理器exit方法。

__enter____exit__方法说明

__enter__方法说明上下文管理器的__enter__方法是可以带返回值的,默认返回None,这个返回值通过with…as…中的 as 赋给它后面的那个变量,所以 with EXPR as VAR 就是将EXPR对象__enter__方法的返回值赋给 VAR当然with...as...并非固定组合,单独使用with...也是可以的,上下文管理器的__enter__方法还是正常执行,只是这个返回值并没有赋给一个变量,with下面的代码块也不能使用这个返回值。__exit__方法说明上下文管理器的__exit__方法接收3个参数exc_typeexc_valexc_tb,如果代码块BLOCK发生了异常e并退出,这3个参数分别为type(e)str(e)e.__traceback__,否则都为None。同样__exit__方法也是可以带返回值的,这个返回值应该是一个布尔类型True或False,默认为None(即False)。如果为False,异常会被抛出,用户需要进行异常处理。如果为True,则表示忽略该异常

类实现with open函数

class File:
  def __init__(self,fileName,mode,encoding="utf-8"):
      self.fileName = fileName
      self.mode = mode
      self.encoding = encoding

  def __enter__(self):
      print("进入")
      self.f = open(self.fileName,self.mode,encoding=self.encoding)
      return self.f

  def __exit__(self, exc_type, exc_val, exc_tb):
      print("退出")
      self.f.close()

with File("test.txt","w") as f:
  f.write("松勤测试")