Python 线程,with的作用(自动获取和释放锁Lock)
import threading
import time
num=0 #全局变量多个线程可以读写,传递数据
mutex=threading.Lock() #创建一个锁
class Mythread(threading.Thread):
def run(self):
global num
with mutex: #with Lock的作用相当于自动获取和释放锁(资源)
for i in range(1000000): #锁定期间,其他线程不可以干活
num+=1
print(num)
mythread=[]
for i in range(5):
t=Mythread()
t.start()
mythread.append(t)
for t in mythread:
t.join()
print("game over")
'''
with mutex: #with表示自动打开自动释放锁
for i in range(1000000): #锁定期间,其他人不可以干活
num+=1
#上面的和下面的是等价的
if mutex.acquire(1):#锁住成功继续干活,没有锁住成功就一直等待,1代表独占
for i in range(1000000): #锁定期间,其他线程不可以干活
num+=1
mutex.release() #释放锁
'''
python的with
with__是从Python2.5引入的一个新的语法,它是一种上下文管理协议,目的在于从流程图中把 try,except 和finally 关键字和资源分配释放相关代码统统去掉,简化__try….except….finlally__的处理流程。with__通过__enter__方法初始化,然后在__exit__中做善后以及处理异常,所以使用__with__处理的对象必须有__enter()和__exit()这两个方法。其中__enter__()方法在语句体(__with__语句包裹起来的代码块)执行之前进入运行,exit()方法在语句体执行完毕退出后运行。with 语句适用于对资源进行访问的场合,确保不管使用过程中是否发生异常都会执行必要的“清理”操作,释放资源,比如文件使用后自动关闭、线程中锁的自动获取和释放等。
With语句的基本语法格式:
with expression [as target]:
with_body
参数说明:
expression
:是一个需要执行的表达式;
target
:是一个变量或者元组,存储的是expression表达式执行返回的结果,可选参数。
with语句的工作原理:
紧跟__with__后面的语句会被求值,返回对象的__enter__()方法被调用,这个方法的返回值将被赋值给as关键字后面的变量,当__with__后面的代码块全部被执行完之后,将调用前面返回对象的__exit__()方法。with语句最关键的地方在于被求值对象必须有__enter__()和__exit__()这两个方法,那我们就可以通过自己实现这两方法来自定义__with__语句处理异常。
示例代码:
#encoding=utf-8
class opened(object):
def __init__(self,filename):
self.handle=open(filename)
print "Resource:%s"%filename
def __enter__(self):
print "[enter%s]: Allocate resource."%self.handle
return self.handle#可以返回不同的对象
def __exit__(self,exc_type,exc_value,exc_trackback):
print "[Exit %s]: Free resource." %self.handle
if exc_trackback is None:
print "[Exit %s]:Exited without exception."%self.handle
self.handle.close()
else:
print "[Exit %s]: Exited with exception raised."%self.handle
return False # 可以省略,缺省的None也是被看做是False
with opened(r'd:\\xxx.txt') as fp:
for line in fp.readlines():
print line
opened中的__enter__() 返回的是自身的引用,这个引用可以赋值给 as 子句中的fp变量;
返回值的类型可以根据实际需要设置为不同的类型,不必是上下文管理器对象本身。
exit() 方法中对变量exc_trackback进行检测,如果不为 None,表示发生了异常,返回 False 表示需要由外部代码逻辑对异常进行处理;
如果没有发生异常,缺省的返回值为 None,在布尔环境中也是被看做 False,但是由于没有异常发生,exit() 的三个参数都为 None,上下文管理代码可以检测这种情况,做正常处理。exit()方法的3个参数,分别代表异常的类型、值、以及堆栈信息。