并发和并行
- 并发:指的是任务数多于CPU核数,通过操作系统的各种任务调度算法,实现多个任务“一起”执行
- 并行:指的是任务数小于CPU核数,即任务真的是一起执行
队列
- python的Queue模块中提供了同步的、线程安全的队列类,包含:FIFO(先入先出)队列queue、LIFO(后入先出)队列LifoQueue,优先级别队列PriorityQueue,这些队列都实现了锁,能够在多线程中直接使用。可以使用队列来实现线程间的同步
- 初始化Queue()对象,如果括号中没有指定最大可接收的消息数量,或者数量为负值,那么久代表可接受对的消息没有上限
队列的方法:
- def task_done(self) :在完成一项工作后,使用task_done()方法可以向队列发送一个消息表示该任务执行完毕
- def join(self) :等待队列中所有的任务执行完毕之后,再往下,否则一直等待
- join()是判断的依据,不单单是列队中没有数据,数据get出去之后,要使用task_done()向队列发送一个信号,表示该任务执行完毕
- def qsize(self) :返回当前队列包含的消息数量
- def empty(self) :如果队列为空,返回True,反之False
- def full(self) :如果队列满了,返回True,反之False
- def put(self,item,block=True,timeout=None) :添加数据
- def get(self,block=True,timeout=None) : 获取队列,timeout设置等待时间
- def put_nowait(self,item)
- def get_nowait(self,item)
from queue import Queue, LifoQueue, PriorityQueue # 实例化一个队列对象 q = Queue() # 队列添加数据:put,put_nowait q.put('python001') q.put('python006', timeout=3) # 参数timeout 设置等待时间 q.put_nowait('pythhn00000') # 添加数据不等待 # get: 获取数据 print(q.get()) print(q.get()) # qsize:获取队列中数据的长度 print(q.qsize()) # 判断队列是否为空 print(q.empty()) # 判断队列是否已满 print(q.full())
from queue import Queue
q=Queue(2)
q.put(1)
q.put(2)
q.task_done()
#q.task_done()
print("join1")
q.join()
print("join2")
"""
只会输出 join1
"""
进程
进程:一个程序运行起来后,代码+用到的资源称为进程,它是操作系统分配资源的基本单位
不仅可以通过现场完成多任务,进程也是可以的
进程的状态
- 工作中,任务数往往大于CPU的核数,即一定有一些任务正在执行,而另一些任务在等待执行
- 就绪状态:运行条件都已经满足,正等在CPU执行
- 执行状态: CPU正在执行其功能
- 等待状态:等待某些条件满足,例如一个程序sleep了,就处于等待状态
window下,执行多进程,程序执行入口需要写在 if __name__=="__main__"函数里
进程、线程对比
功能
- 进程,能够完成多任务,比如 在一台电脑上能够同时运行多个软件
- 线程, 能够完成多任务,比如 一个QQ中的多个聊天窗口
定义的不同
- 进程是系统进行资源分配的单位
- 线程是进程的一个实体,是CPU调度和任务分派的基本单位,它是比进程更小的能独立运行的基本单位,线程自己基本上不拥有系统资源,只拥有一点在运行中必不可少的资源,但是她可与同属一个进程的其他线程共享进程所拥有的全部资源
queue.Queue:只能在同一个进程中的多个线程之间使用
multiprocessing.Queue: 可以在多个进程之间跨进程输出数据
进程池
- 当需要创建的子进程数量不多时,可以直接利用multiprocessing中的Process动态生成多个进程,但如果是上百甚至上千个目标,手动去创建进程的工作量巨大,这个时候就可以用到该模块提供的Pool方法
- 初始化Pool时,可以指定一个最大进程数,当有新的请求提交到Pool中,如果池还没有满,那么就有创建一个新的进程用来执行该请求;但如果池中的进程数已经达到指定的最大值,那么请求就会等待,直到池中有进程结束,才会有之前的进程来执行新的任务
Pool常用方法:
- apply_async(func[,args[,kwds]]) :使用非阻塞方式调用func(并行执行,堵塞方式必须等待上一个进程退出才能执行下一个进程),args为传递给func的参数列表,kwds为传递给func的关键字参数列表
- close() :关闭Pool,使其不再接受新的任务
- join() : 主进程堵塞,等待子进程的退出,必须在close之后使用
进程池的Queue
- 如果要使用Pool创建进程,就需要使用multiprocessing.Manager()中的Queue(),而不是multiprocessing。Queue(),否则就会报错
from multiprocessing import Pool import os import time import random def worker(msg): t_start = time.time() print("%s开始执行,进程号为%d" % (msg, os.getpid())) # random.random()随机生成0~1之间的浮点数 time.sleep(random.random() * 2) t_stop = time.time() print(msg, "执行完毕,耗时%0.2f" % (t_stop - t_start)) if __name__ == '__main__': po = Pool(3) # 定义一个进程池,最大进程数3 for i in range(0, 10): # Pool().apply_async(要调用的目标,(传递给目标的参数元祖,)) # 每次循环将会用空闲出来的子进程去调用目标 po.apply_async(worker, (i,)) # print("----start----") po.close() # 关闭进程池,关闭后po不再接收新的请求 po.join() # 等待po中所有子进程执行完成,必须放在close语句之后 print("-----end-----")
协程
协程:又称微线程,是python中另外一种实现多任务的方式,只不过比线程更小占用更小执行单元。因为它自带CPU上下文,这样只要在合适gr的时机,可以把一个协程 切换到另一个协程。只要这个过程中保存或者恢复CPU上下文那么程序还是可以运行的。
通俗的描述:
- 协程是线程中的一个特殊的函数,这个函数执行的时候,可以在某个地方暂停,并且可以重新在暂停处,继续运行,协程在进行切换的时候,只需要保存当前协程函数中的一些临时变量等信息,然后切换到另外一个函数中运行,并且切换的次数以及什么时候再切换到原来的函数,都可以由开发者自己决定。
- 协程切换的时候,既不涉及资源切换,也不涉及操作系统的调度,而是在同一个程序中切换不同的函数执行,所以协程占用的资源非常少,一秒钟可以切换上百万次系统
- 协程与进程、线程相比并不是一个维度的概念
gevent模块
原理:当一个协程遇到IO操作时,比如访问网络,就会自动切换到其他greenlet,等到IO操作完成,再在合适的时候切换回来继续执行。由于IO操作非常耗时,经常使程序处于等待状态,有了gevent就会自动切换协程,就能保证总有greenlei在运行,而不是等待IO
程序补丁:monkey.patch_all()
打补丁作用,只要有耗时的地方,自动切换任务,不局限于gevent.sleep()
注意点:只能在单线程中使用,不能在多线程中使用
import gevent
from gevent import monkey
monkey.patch_all()
import time from gevent.monkey import patch_all patch_all() import gevent import requests def work1(name): for i in range(10): print('这个是work1', name) time.sleep(0.1) def work2(): for i in range(10): print('这个是work2') time.sleep(0.1) g1 = gevent.spawn(work1, 'aa') g2 = gevent.spawn(work2) g1.join() g2.join()