python线程与进程小结
传统方式是调用2个方法执行1个任务,方法按顺序依次执行
# -*- coding:utf-8 -*- import threading import time def run(n): print('task',n) time.sleep(3) if __name__ == '__main__': run('t1') run('t2')
多线程例子
2个线程同时并发执行1个任务
# -*- coding:utf-8 -*- import threading import time def run(n): print('task',n) time.sleep(3) if __name__ == '__main__': t1=threading.thread(target=run,args=('t1',)) t2=threading.thread(target=run,args=('t2',)) t1.start() t2.start()
自己写一个类继承继承threading.thread
# -*- coding:utf-8 -*- import threading import time class mythread(threading.thread): def __init__(self,n): super(mythread,self).__init__() self.n=n #这里面默认就有一个run方法 def run(self): print('runing task',self.n) if __name__ == '__main__': #在主方法通过对象调用线程 t1=mythread('t1') t2=mythread('t2') t1.run() t2.run()
使用for循环启动多个线程
# -*- coding:utf-8 -*- import threading import time def run(n): print('task',n) time.sleep(3) if __name__ == '__main__': for i in range(10): t=threading.thread(target=run,args=('t-%s'%i,)) t.start()
等多线程同时执行完后,再执行其它代码,因为线程是与其它代码一起运行的
# -*- coding:utf-8 -*- import threading import time def run(n): print('task',n) time.sleep(3) if __name__ == '__main__': time_start=time.time() #定义一个空列表装线程t实例 t_objects=[] for i in range(10): t=threading.thread(target=run,args=('t-%s'%i,)) t.start() t_objects.append(t) # 等所有线程执行完后,再执行下面的代码 # 因为线程与下面的代码是同时运行的 # 要想先等线程执行完毕再执行其它代码 # 使用join()方法进行阻塞 #在这里统一结束所有t线程 for i in t_objects: t.join() time_end=time.time() sun=time_end-time_start print(sun)
守护线程
程序会等非守护线程执行完毕,才会结束程序,不会等守护线程执行完毕
# -*- coding:utf-8 -*- import threading import time def run(n): print('task',n) time.sleep(3) if __name__ == '__main__': time_start=time.time() #定义一个空列表装线程t实例 # t_objects=[] for i in range(10): t=threading.thread(target=run,args=('t-%s'%i,)) t.setdaemon(true)#把当前所有的子线程设置为守护线程 t.start() # t_objects.append(t) # for i in t_objects: # t.join() time_end=time.time() sun=time_end-time_start print(sun) # 主线程退出时不会等待子线程执行完毕后再退出 print('主线程退出')
# -*- coding:utf-8 -*- __author__ = "alex li" import threading import time def run(n): lock.acquire() global num num +=1 time.sleep(1) lock.release() lock = threading.lock() num = 0 t_objs = [] #存线程实例 for i in range(10): t = threading.thread(target=run,args=("t-%s" %i ,)) t.start() t_objs.append(t) #为了不阻塞后面线程的启动,不在这里join,先放到一个列表里 for t in t_objs: #循环线程实例列表,等待所有线程执行完毕 t.join() print("----------all threads has finished...",threading.current_thread(),threading.active_count()) print("num:",num)
import time import threading def addnum(): global num #在每个线程中都获取这个全局变量 print('--get num:',num ) time.sleep(1) num -=1 #对此公共变量进行-1操作 num = 100 #设定一个共享变量 thread_list = [] for i in range(100): t = threading.thread(target=addnum) t.start() thread_list.append(t) for t in thread_list: #等待所有线程执行完毕 t.join() print('final num:', num )
正常来讲,这个num结果应该是0, 但在python 2.7上多运行几次,会发现,最后打印出来的num结果不总是0,为什么每次运行的结果不一样呢? 哈,很简单,假设你有a,b两个线程,此时都 要对num 进行减1操作, 由于2个线程是并发同时运行的,所以2个线程很有可能同时拿走了num=100这个初始变量交给cpu去运算,当a线程去处完的结果是99,但此时b线程运算完的结果也是99,两个线程同时cpu运算的结果再赋值给num变量后,结果就都是99。那怎么办呢? 很简单,每个线程在要修改公共数据时,为了避免自己在还没改完的时候别人也来修改此数据,可以给这个数据加一把锁, 这样其它线程想修改此数据时就必须等待你修改完毕并把锁释放掉后才能再访问此数据。
*注:不要在3.x上运行,不知为什么,3.x上的结果总是正确的,可能是自动加了锁
加锁版本
import time import threading def addnum(): global num #在每个线程中都获取这个全局变量 print('--get num:',num ) time.sleep(1) lock.acquire() #修改数据前加锁 num -=1 #对此公共变量进行-1操作 lock.release() #修改后释放 num = 100 #设定一个共享变量 thread_list = [] lock = threading.lock() #生成全局锁 for i in range(100): t = threading.thread(target=addnum) t.start() thread_list.append(t) for t in thread_list: #等待所有线程执行完毕 t.join() print('final num:', num )
gil vs lock
机智的同学可能会问到这个问题,就是既然你之前说过了,python已经有一个gil来保证同一时间只能有一个线程来执行了,为什么这里还需要lock? 注意啦,这里的lock是用户级的lock,跟那个gil没关系 ,具体我们通过下图来看一下+配合我现场讲给大家,就明白了。
那你又问了, 既然用户程序已经自己有锁了,那为什么c python还需要gil呢?加入gil主要的原因是为了降低程序的开发的复杂度,比如现在的你写python不需要关心内存回收的问题,因为python解释器帮你自动定期进行内存回收,你可以理解为python解释器里有一个独立的线程,每过一段时间它起wake up做一次全局轮询看看哪些内存数据是可以被清空的,此时你自己的程序 里的线程和 py解释器自己的线程是并发运行的,假设你的线程删除了一个变量,py解释器的垃圾回收线程在清空这个变量的过程中的clearing时刻,可能一个其它线程正好又重新给这个还没来及得清空的内存空间赋值了,结果就有可能新赋值的数据被删除了,为了解决类似的问题,python解释器简单粗暴的加了锁,即当一个线程运行时,其它人都不能动,这样就解决了上述的问题, 这可以说是python早期版本的遗留问题。
rlock(递归锁)
说白了就是在一个大锁中还要再包含子锁
import threading,time def run1(): print("grab the first part data") lock.acquire() global num num +=1 lock.release() return num def run2(): print("grab the second part data") lock.acquire() global num2 num2+=1 lock.release() return num2 def run3(): lock.acquire() res = run1() print('--------between run1 and run2-----') res2 = run2() lock.release() print(res,res2) if __name__ == '__main__': num,num2 = 0,0 lock = threading.rlock() for i in range(10): t = threading.thread(target=run3) t.start() while threading.active_count() != 1: print(threading.active_count()) else: print('----all threads done---') print(num,num2)
进程