Python中的with是测试常用到的资源打开利

时间:2022-07-25
本文章向大家介绍Python中的with是测试常用到的资源打开利,主要内容包括其使用实例、应用技巧、基本知识点总结和需要注意事项,具有一定的参考价值,需要的朋友可以参考一下。

“在我们完成自动化测试代码的时候,总会遇见各种读取文本文件、读取Excel等类型的操作。这种代码我们时刻都要记得打开文件后要关闭文件。往往关闭文件却是我们常常忘记的。针对上述这样的情况,python提供了with就可以完美解决这个问题,这也是python的语法糖。Syntactic sugar,也就是语法糖,它指计算机语言中添加的某种语法,这种语法对语言的功能没有影响,但是更方便程序员使用。语法糖让程序更加简洁,有更高的可读性。语法糖就是为了避免coder出现错误并提高效率的语法层面的一种优雅的解决方案。 ”

一个常规的文件打开代码

下面是一个常规的打开文件的代码,那么你可以从下面代码中看出什么问题吗?

rf= open('crisschan.txt','r')
print(rf.readlines())
rf.close()

上面代码在读取文件过程中如果发生异常,那么close()函数就没有办法被执行到了,这也就导致了文件没有办法关闭了。因此,很多教程上都告诉大家要用try except将文件读取的异常捕获到,那么我们改造一下如下:

try:
    rf = open('email.txt','r')
    print(rf.readlines())

except:
    print('Oooooops!What's up!')
finally:
    rf.close()

好了,上面的代码无论如何我们都会执行close函数了,这样是不是已经很好了。但是上面的代码太繁琐了,这样的coding段,python提供了with,让上述的代码更简单:

with open('email.txt','r') as rf:
    print(rf.readlines())

上面是不是很简洁,是不是也很优雅呢。

with是怎么干活的

with context_expression [as target(s)]:
        do_somthing()

上述代码片段中,context_expression会返回一个上下文管理器对象,这个对象并不赋值给as后的target(s),而是上下文管理器的__enter__()函数的返回值赋值给 target(s)。当with全部的代码段都执行完成后,会调用__exit__()

具体例子如下:

class SampleWith(object):
    def __init__(self):
        print('init this class')
    def __enter__(self):
        print('this is __enter__')
        return 'CrissChan'
    def __exit__(self, exc_type, exc_val, exc_tb):
        print('this is __exit__')
    def call(self):
        print('call funtion')

if __name__ == '__main__':
    with SampleWith() as sw:
        print('sw is :',sw)
        sw.call()

运行完后,输入如下:

init this class
this is __enter__
sw is : CrissChan
this is __exit__
Traceback (most recent call last):
File "/Users/crisschan/PycharmProjects/try_space/flv2mp4.py", line 24, in <module>
    sw.call()
AttributeError: 'str' object has no attribute 'call'

那下面我来给你解释一下上面的代码段以及结果输出。

  • 1、在main函数中我们使用with调用了SampleWith(),这时候我们就会看到了,我们先实例化了一个SampleWith类,调用了他的__init__(self)构造函数,
  • 2、接下来因为我们使用了with这个语法糖,因此下面调用了__enter__(self)
  • 3、在后面我使用了as sw,也就是我将__enter__(self)的return赋值给了sw,那么也就是说sw存储的是字符串CrissChan,那么也就打印了sw is : CrissChan
  • 4、后面调用了sw.call()就出了问题,因为字符串没有call方法。但是仍旧进入了__exit__(self, exc_type, exc_val, exc_tb)函数。

下面我将上面有报错的代码修改一下,如下:

class SampleWith(object):
    def __init__(self):
        print('init this class')
    def __enter__(self):
        print('this is __enter__')
        return 'CrissChan'
    def __exit__(self, exc_type, exc_val, exc_tb):
        print('this is __exit__')
    def call(self):
        print('call funtion')

if __name__ == '__main__':
    with SampleWith() as sw:
        print('sw is :',sw)

上面代码就不会报错了。从上面可以看出就算中间除了异常,放在__exit__()中的逻辑段还是会被执行的。想要了解更多请看官方的文档pep-403

特别备注:

exit()方法中有3个参数, exc_type, exc_val, exc_tb,这些参数在异常处理中相当有用。
exc_type: 错误的类型
exc_val: 错误类型对应的值
exc_tb: 代码中错误发生的位置