9.11 进程池与线程池

池子使用来限制并发的任务数目,限制我们的计算机在一个自己可承受的范围内去并发地执行任务

池子内什么时候装进程:并发的任务属于计算密集型池子内什么时候装线程:并发的任务属于IO密集型

进程池:

from concurrent.futures importProcessPoolExecutor,ThreadPoolExecutorimporttime,os,random

​deftask(x):print('%s 接客' %os.getpid())

time.sleep(random.randint(2,5))return x**2​if __name__ == '__main__': #ProcessPoolExecutor创建并开启指定数目的进程

p=ProcessPoolExecutor() #默认开启的进程数是cpu的核数

​for i in range(20):

p.submit(task,i)#一下并行执行四个任务,等其中一个任务执行完后再执行下一个

线程池:

from concurrent.futures importProcessPoolExecutor,ThreadPoolExecutorimporttime,os,random

​deftask(x):print('%s 接客' %x)

time.sleep(random.randint(2,5))return x**2​if __name__ == '__main__': #ThreadPoolExecutor创建并开启指定数目的线程

p=ThreadPoolExecutor(4) #默认开启的线程数是cpu的核数*5

​for i in range(20):

p.submit(task,i)#一下并发执行四个任务,等其中一个任务执行完后再并发执行下一个

9.112 基于多线程实现并发的套接字通信(使用线程池)

服务端:

from socket import *

from threading importThreadfrom concurrent.futures importProcessPoolExecutor,ThreadPoolExecutor

tpool=ThreadPoolExecutor(3) #ThreadPoolExecutor创建并开启指定数目的线程

defcommunicate(conn,client_addr):while True: #通讯循环

try:

data= conn.recv(1024)if not data: breakconn.send(data.upper())exceptConnectionResetError:breakconn.close()

​defserver():

server=socket(AF_INET,SOCK_STREAM)

server.bind(('127.0.0.1',8080))

server.listen(5)

​while True: #链接循环

conn,client_addr=server.accept()print(client_addr)#t=Thread(target=communicate,args=(conn,client_addr))

#t.start()

tpool.submit(communicate,conn,client_addr)#一下并发执行3个任务,等其中一个任务执行完后再并发执行下一个

server.close()

​if __name__ == '__main__':

server()

View Code

客户端:

from socket import *client=socket(AF_INET,SOCK_STREAM)

client.connect(('127.0.0.1',8080))

​whileTrue:

msg=input('>>>:').strip()if not msg:continueclient.send(msg.encode('utf-8'))

data=client.recv(1024)print(data.decode('utf-8'))

client.close()

View Code

9.12 同步异步阻塞非阻塞

阻塞与非阻塞指的是程序的两种运行状态:

阻塞:遇到 I/O 就发生阻塞,程序一旦遇到阻塞操作就会停在原地,并且立刻释放CPU资源

非阻塞(就绪态或运行态):没有遇到 I/O 操作,或者通过某种手段让程序即便是遇到 I/O 操作也不会停在原地,执行其他操作,力求尽可能多的占有CPU

同步与异步指的是提交任务的两种方式:

同步调用:提交完任务后,就在原地等待,直到任务运行完毕后,拿到任务的返回值,才继续执行下一行代码

异步调用:提交完任务后,不在原地等待,直接执行下一行代码

from concurrent.futures importProcessPoolExecutor,ThreadPoolExecutorimporttime,os,random#from multiprocessing import Pool

deftask(x):print('%s 接客' %x)

time.sleep(random.randint(1,3))return x**2​if __name__ == '__main__':#异步调用

p=ThreadPoolExecutor(4) #默认开启的线程数是cpu的核数*5

obj_l=[]for i in range(10):

obj=p.submit(task,i)

obj_l.append(obj)

​#p.close()

#p.join()

p.shutdown(wait=True)#shutdown指的是不能再往进程池内提交任务,wait=True指等待进程池或线程池内所有的任务都运行完毕

print(obj_l[3].result()) #9 #最后拿结果

print('主')

​#同步调用

p=ThreadPoolExecutor(4) #默认开启的线程数是cpu的核数*5

for i in range(10):print(p.submit(task,i).result())print('主')

9.121 异步调用+回调机制

问题:

1、任务的返回值不能得到及时的处理,必须等到所有任务都运行完毕才能统一进行处理

2、解析的过程是串行执行的,如果解析一次需要花费2s,解析9次则需要花费18s

基于进程池:

from concurrent.futures importProcessPoolExecutor,ThreadPoolExecutorimportrequestsimportosimporttimeimportrandom

​defget(url):print('%s GET %s' %(os.getpid(),url))

response=requests.get(url)

time.sleep(random.randint(1,3))if response.status_code == 200:returnresponse.text

​def pasrse(obj): #干解析的活

res=obj.result() #回调拿结果

print('%s 解析结果为:%s' %(os.getpid(),len(res))) #4108 解析结果为:2443

​if __name__ == '__main__':

urls=['https://www.baidu.com','https://www.baidu.com','https://www.baidu.com','https://www.baidu.com','https://www.baidu.com','https://www.baidu.com','https://www.baidu.com','https://www.baidu.com','https://www.python.org',

]

pool=ProcessPoolExecutor(4)for url inurls:

obj=pool.submit(get,url) #parse函数会在obj对应的任务执行完毕后自动执行,会把obj自动传给parse

obj.add_done_callback(pasrse) #四个进程并发爬取信息,主进程在执行解析操作

​print('主进程',os.getpid()) #主进程 4108

View Code

基于线程池:

from concurrent.futures importProcessPoolExecutor,ThreadPoolExecutorfrom threading importcurrent_threadimportrequestsimportosimporttimeimportrandom

​defget(url):print('%s GET %s' %(current_thread().name,url))

response=requests.get(url)

time.sleep(random.randint(1,3))if response.status_code == 200:returnresponse.text

​def pasrse(obj): #干解析的活

res=obj.result()print('%s 解析结果为:%s' %(current_thread().name,len(res)))#ThreadPoolExecutor-0_1 解析结果为:

#2443

if __name__ == '__main__': #ThreadPoolExecutor-0_3 解析结果为:2443

urls=['https://www.baidu.com','https://www.baidu.com','https://www.baidu.com','https://www.baidu.com','https://www.baidu.com','https://www.baidu.com','https://www.baidu.com','https://www.baidu.com','https://www.python.org',

]

pool=ThreadPoolExecutor(4)for url inurls:

obj=pool.submit(get,url) #parse函数会在obj对应的任务执行完毕后自动执行,会把obj自动传给parse

obj.add_done_callback(pasrse) #四个线程并发爬取信息,空闲者执行解析操作

print('主线程',current_thread().name) #主线程 MainThread

View Code

9.13 线程queue

队列:先进先出 queue.Queue()

importqueue

q=queue.Queue(3)

q.put(1)

q.put(2)

q.put(3)#q.put(4) 阻塞

​print(q.get()) #1

print(q.get()) #2

print(q.get()) #3

堆栈:后进先出 queue.LifoQueue()

importqueue

q=queue.LifoQueue(3)

q.put('a')

q.put('b')

q.put('c')

​print(q.get()) #c

print(q.get()) #b

print(q.get()) #a

优先级队列:可以以小元组的形式往队列里存值,第一个元素代表优先级,数字越小优先级越高

PriorityQueue()

importqueue

q=queue.PriorityQueue(3)

q.put((10,'user1'))

q.put((-3,'user2'))

q.put((-2,'user3'))

​print(q.get()) #(-3, 'user2')

print(q.get()) #(-2, 'user3')

print(q.get()) #(10, 'user1')

python time sleep 阻塞 异步_python 之 并发编程(进程池与线程池、同步异步阻塞非阻塞、线程queue)...相关推荐

  1. 学习笔记(33):Python网络编程并发编程-进程池线程池

    立即学习:https://edu.csdn.net/course/play/24458/296451?utm_source=blogtoedu 进程池与线程池: 一般应用在网站上,进程池或线程池最大的 ...

  2. 15分钟读懂进程线程、同步异步、阻塞非阻塞、并发并行,太实用了!

    作者:Martin cnblogs.com/mhq-martin/p/9035640.html 基本概念 1 进程和线程 进程(Process): 是Windows系统中的一个基本概念,它包含着一个运 ...

  3. python并发编程之semaphore(信号量)_python 之 并发编程(守护进程、互斥锁、IPC通信机制)...

    9.5 守护进程 主进程创建守护进程 其一:守护进程会在主进程代码执行结束后就立即终止 其二:守护进程内无法再开启子进程,否则抛出异常:AssertionError: daemonic process ...

  4. 15分钟读懂进程线程、同步异步、阻塞非阻塞、并发并行

    基本概念 1 进程和线程 进程(Process): 是Windows系统中的一个基本概念,它包含着一个运行程序所需要的资源.一个正在运行的应用程序在操作系统中被视为一个进程,进程可以包括一个或多个线程 ...

  5. 进程线程、同步异步、阻塞非阻塞、并发并行、多线程

    一: 进程和线程 1: 进程(Process) 是Windows系统中的一个基本概念,它包含着一个运行程序所需要的资源.一个正在运行的应用程序在操作系统中被视为一个进程,进程可以包括一个或多个线程.线 ...

  6. 深入理解非阻塞同步IO和非阻塞异步IO

    这两篇文章分析了Linux下的5种IO模型 http://blog.csdn.net/historyasamirror/article/details/5778378 http://blog.csdn ...

  7. Java并发编程进阶——多线程的安全与同步

    多线程的安全与同步 多线程的操作原则 多线程 AVO 原则 A:即 Atomic,原子性操作原则.对基本数据类型变量的读和写是保证原子性的,要么都成功,要么都失败,这些操作不可中断. V:即 vola ...

  8. python 进程池阻塞和非阻塞_python 之 并发编程(进程池与线程池、同步异步阻塞非阻塞、线程queue)...

    9.11 进程池与线程池 池子使用来限制并发的任务数目,限制我们的计算机在一个自己可承受的范围内去并发地执行任务 池子内什么时候装进程:并发的任务属于计算密集型池子内什么时候装线程:并发的任务属于IO ...

  9. python多线程执行其他模块的文件_python并发编程--进程线程--其他模块-从菜鸟到老鸟(三)...

    concurrent模块 1.concurrent模块的介绍 concurrent.futures模块提供了高度封装的异步调用接口 ThreadPoolExecutor:线程池,提供异步调用 Proc ...

  10. python创建新进程_Python并发编程(进程的创建)

    动态性:进程的实质是程序在多道程序系统中的一次执行过程,进程是动态产生,动态消亡的. 并发性:任何进程都可以同其他进程一起并发执行 独立性:进程是一个能独立运行的基本单位,同时也是系统分配资源和调度的 ...

最新文章

  1. api网关选型_如何轻松打造百亿流量API网关?看这一篇就够了(下)
  2. TensorFow的基本使用
  3. golang 协程同步 简介
  4. 微信与服务器通讯失败,linux服务器微擎提示couldn’t resolve host api.weixin.qq.com解决办法...
  5. Sphinx-安装和配置
  6. 用hundred造句子_关于冬至的问候短句,冬至文案唯美句子
  7. 按键精灵调用迅雷下载文件
  8. 马尔可夫链 (Markov Chain)是什么鬼
  9. 辨别 Python 中 load 和 loads 的小技巧
  10. HTTPS 跟 HTTP区别简述
  11. 在centos7安装nodejs并升级nodejs到最新版本
  12. 数据/方法论固然重要,但人为分析更有价值!
  13. 车牌识别及验证码识别的一般思路
  14. 计算机科学与技术的主要研究方向,计算机科学与技术学科方向与特色
  15. 基于java 海康视频监控 jar包运行
  16. plc和pc串口通讯接线_电脑和PLC连接不上我用的是USB转串口的连接线
  17. HTML5拖放(drag和drog)
  18. 东沃电子:瞬态电压抑制二极管选型指南
  19. css 实现一个尖角_css 实现一个带尖角的正方形
  20. flask 图文混排

热门文章

  1. [网络流24题] 试题库问题
  2. 归并排序(包含逆序数对的个数51Nod1019)
  3. JavaScript的注意事项
  4. LightOj 1088 - Points in Segments (二分枚举)
  5. CMake实践(2)
  6. Java final关键字详解
  7. Dubbo负载均衡算法初步解析
  8. 【Hoxton.SR1版本】Spring Cloud Gateway之GlobalFilter全局过滤器
  9. 自定义类加载器的父类为什么是AppClassLoader?
  10. List集合之LinkedList