一。线程池

  线程池是一个处理线程任务的集合,他是可以接受一定量的线程任务,并创建线程,处理该任务,处理结束后不会立刻关闭池子,会继续等待提交的任务,也就是他们的进程/线程号不会改变。

  当线程池中的任务没有结束时是不会接受下一个任务的。

  它的操作有:

  pool = ThreadPoolExecutor()

  创建一个线程池,其中括号中代表的是一次可以接纳的线程任务,可以不加参数,不加参数其数量就是当前cpu的个数*5。

  res = pool.submit(func,args)

  提交一个任务,args代表的是函数的参数。res接受的是该submit的返回值,类似于如下的类:

<Future at 0x2057e656940 state=running>

  state代表的是当前该线程的状态。

  res.result()

  而使用result可以将提交的任务函数的返回值获取。

  这里的result还有等待任务的返回值的作用。如果任务没结束,就会一直等待,可以将并行操作改成串行操作。

  pool.shutdown()

  可以将池子关闭,并等待池子终端 任务全部结束再执行下面代码。  

例子:

import time
from concurrent.futures import ThreadPoolExecutor
import os
from gevent import os pool = ThreadPoolExecutor(5) def task(n):
print(n,os.getpid())
time.sleep(2)
return n**2 list_1 = []
for i in range(20):
res = pool.submit(task,i) #提交任务
print(res.result())#等待任务的返回值
list_1.append(res) pool.shutdown() #关闭池子,等待池子中的任务运行完毕
for j in list_1:
print('>>>',j.result())
print('主')

  进程池:

  进程池的使用和线程池差不多,区别仅只有包名不同,在进程池中我们可以验证以下池中的进程/线程是否是用的同样的进程/线程,使用os。getpid()方法即可。

  进程值不传值,里面的数值默认时cpu的个数。

import time
from concurrent.futures import ThreadPoolExecutor,ProcessPoolExecutor
import os
from gevent import os # pool = ThreadPoolExecutor(5)
pool = ProcessPoolExecutor(5) def task(n):
print(n,os.getpid())
time.sleep(2)
return n**2 def callback(n):
print(n.result()) if __name__ == '__main__':
list_1 = []
for i in range(20):
res = pool.submit(task,i).add_done_callback(callback) #提交任务
# print(res.result())#等待任务的返回值
list_1.append(res)

  异步回调:

  除了上面使用的将返回的future对象添加到列表,再调用result()方法返回其返回值以外,还可以对指派任务的返回值用

add_done_callback(callback)

  方法,将该对象回调到callback(可以自定义)函数,由函数接纳处理该值,函数的参数就是任务的返回值,多个返回值要设置多个参数。

  回调是在生产返回值时就运行的。

二。协程

  协程就是在单线程的情况下实现并发。

  一般程序的多道技术都是用 切换+保存状态 实现

  在一般的cpu运算时,都是在五种状态中来回切换的,程序运行的5种状态:

  1.新建。2.就绪。3.运行。4.阻塞。5.结束。

  一般的,程序都是在2,3,4的状态来回切换,有2种情况。

  1,程序遇到了io操作,由运行态进入到了阻塞态,直到io操作结束后再到阻塞态等待时间片。

  2.程序的时间片用完,由运行态到就绪态。

  协程的作用就是使得线程遇到io操作自己切换,运行的方式从1.变成2.线程持续不断的就绪,可以获得大量的cpu运算时间。

  要实现这个功能需要考虑线程的保存状态问题。

  这里就要用到迭代器的知识,yield,

  yield可以保存上一次操作的状态,所以使用yield可以验证协程对计算密集型的线程操作后是否能加快效率。

#串行执行 0.8540799617767334
# import time
#
# def func1():
# for i in range(10000000):
# i+1
#
# def func2():
# for i in range(10000000):
# i+1
#
# start = time.time()
# func1()
# func2()
# stop = time.time()
# print(stop - start)
#基于yield并发执行  1.3952205181121826
# import time
# def func1():
# while True:
# 10000000+1
# yield
#
# def func2():
# g=func1()
# for i in range(10000000):
# time.sleep(100) # 模拟IO,yield并不会捕捉到并自动切换
# i+1
# next(g)
#
# start=time.time()
# func2()
# stop=time.time()
# print(stop-start)

  可以看到,在计算密集的线程中,不断切换线程是不利于程序的运行的。

  而yield不能识别io操作,而进行线程之间的切换的,所以需要引入一个模块gevent。

  gevent是一个可以识别io的魔块,但不能识别time.sleep,所以还要调用另外一个模块识别time.sleep。

from gevent import monkey;monkey.patch_all() 
# 由于该模块经常被使用 所以建议写成一行
from gevent import spawn
import time

  spawn()可以检测()中的所有任务

def heng():
print("哼")
time.sleep(2)
print('哼') def ha():
print('哈')
time.sleep(3)
print('哈') def heiheihei():
print('嘿嘿嘿')
time.sleep(5)
print('嘿嘿嘿') start = time.time()
g1 = spawn(heng)
g2 = spawn(ha) # spawn会检测所有的任务
g3 = spawn(heiheihei)
g1.join()
g2.join()
g3.join()
# heng()
# ha()
print(time.time() - start)


嘿嘿嘿


嘿嘿嘿
5.033252716064453

  原本10秒钟的程序,现在需要5秒钟就可以运行结束了。

  spawn可以将所有线程添加至一个列表,轮流运行其没有io操作的部分。

  spawn有一个返回值g

  注意,需要在程序最后等待所有程序都运行结束才结束程序,使用g.join方法。

三。使用gevent实现tcp的并发

from gevent import monkey;monkey.patch_all()
import socket
from gevent import spawn server = socket.socket()
server.bind(('127.0.0.1',8080))
server.listen(5) def talk(conn):
while True:
try:
data = conn.recv(1024)
if len(data) == 0:break
print(data.decode('utf-8'))
conn.send(data.upper())
except ConnectionResetError as e:
print(e)
break
conn.close() def server1():
while True:
conn, addr = server.accept()
spawn(talk,conn) if __name__ == '__main__':
g1 = spawn(server1)
g1.join()

四。IO模型。

  1.阻塞型IO

  阻塞型io是在进行io操作时,先跳入阻塞态,然后等待数据。

  数据获得后拷贝数据,

  最后再进入就绪态,

  其中等待数据和拷贝数据都是再阻塞状态:

  2.非阻塞io

  非阻塞io是在遇到io操作时,先发送接受数据请求,如果没有数据就返回一个没有的信号,之后会反复发送数据请求,直到有数据为止,这种模型很占cpu操作。

  3.IO多路复用

  这个模型中有一个select,是一个监测机制,类似于列表,管理io操作。

  当需要进行io操作时,调用select寻找数据,如果找到数据就返回数据,

  等待的操作全部交给select。

  4.异步IO(asyn。。。。)

  在遇到io操作时,有一个回调机制,当需要io操作时,回调机制(内存中)会去寻找数据,当寻找到数据后会返回数据

day33_8_15 并发编程4,线程池与协程,io模型的更多相关文章

  1. python 之 并发编程(线程Event、协程)

    9.14 线程Event connect线程执行到event.wait()时开始等待,直到check线程执行event.set()后立即继续线程connect from threading impor ...

  2. 8.15 day33 进程池与线程池_协程_IO模型(了解)

    进程池和线程池 开进程开线程都需要消耗资源,只不过两者比较的情况线程消耗的资源比较少 在计算机能够承受范围之内最大限度的利用计算机 什么是池? ​ 在保证计算机硬件安全的情况下最大限度地利用计算机 ​ ...

  3. 并发编程(六)——进程/线程池、协程、gevent第三方库

    进程/线程池.协程.gevent第三方库 一.进程/线程池 1.进程池 (1)什么是进程池 如果需要创建的子进程数量不大,可以直接利用multiprocess中的Process来创建.但是当需要创建上 ...

  4. 并发编程 --进、线程池、协程、IO模型

    内容目录: 1.socket服务端实现并发 2.进程池,线程池 3.协程 4.IO模型 1.socket服务端实现并发 # 客户端: import socket client = socket.soc ...

  5. Java并发编程:线程池的使用

    Java并发编程:线程池的使用 在前面的文章中,我们使用线程的时候就去创建一个线程,这样实现起来非常简便,但是就会有一个问题: 如果并发的线程数量很多,并且每个线程都是执行一个时间很短的任务就结束了, ...

  6. 并发编程 13—— 线程池的使用 之 配置ThreadPoolExecutor 和 饱和策略

    Java并发编程实践 目录 并发编程 01—— ThreadLocal 并发编程 02—— ConcurrentHashMap 并发编程 03—— 阻塞队列和生产者-消费者模式 并发编程 04—— 闭 ...

  7. Java并发编程:线程池的使用(转)

    Java并发编程:线程池的使用 在前面的文章中,我们使用线程的时候就去创建一个线程,这样实现起来非常简便,但是就会有一个问题: 如果并发的线程数量很多,并且每个线程都是执行一个时间很短的任务就结束了, ...

  8. (转)Java并发编程:线程池的使用

    背景:线程池在面试时候经常遇到,反复出现的问题就是理解不深入,不能做到游刃有余.所以这篇博客是要深入总结线程池的使用. ThreadPoolExecutor的继承关系 线程池的原理 1.线程池状态(4 ...

  9. Java并发编程:线程池的使用(转载)

    转载自:https://www.cnblogs.com/dolphin0520/p/3932921.html Java并发编程:线程池的使用 在前面的文章中,我们使用线程的时候就去创建一个线程,这样实 ...

  10. Java并发编程:线程池的使用(转载)

    文章出处:http://www.cnblogs.com/dolphin0520/p/3932921.html Java并发编程:线程池的使用 在前面的文章中,我们使用线程的时候就去创建一个线程,这样实 ...

随机推荐

  1. luoguP3181 [HAOI2016]找相同字符

    题意 考虑将\(s1\)和\(s2\)接在一起求出相同子串个数,再求出\(s1\)自己匹配的相同子串个数和\(s2\)自己匹配的相同子串个数减去即可. 如何求相同子串个数: 我们知道子串的集合即所有后 ...

  2. Django middleware (中间件)

    关于中间价: django 中的中间件其实就是一个类,在请求到来和结束后,django会根据自己的规则在合适的时机执行中间件中相应的方法. 在django项目的settings中,有一个 MIDDLE ...

  3. MySQL实战45讲学习笔记:第三十三讲

    一.引子 我经常会被问到这样一个问题:我的主机内存只有 100G,现在要对一个 200G 的大表做全表扫描,会不会把数据库主机的内存用光了? 这个问题确实值得担心,被系统 OOM(out of mem ...

  4. 消息队列的使用<一>:介绍、使用场景和JMS概念知识

    目录 介绍 消息队列的理解 举个栗子 使用场景 消息队列的模型与概念理解 JMS模型 基本概念: 内容: JMS定义的消息结构: PTP式消息传递 PUB/SUB式消息传递 可靠性机制 事务 消息持久 ...

  5. c# lock 锁

    lock语句 lock 语句获取给定对象的互斥 lock,执行语句块,然后释放 lock. 持有 lock 时,持有 lock 的线程可以再次获取并释放 lock. 阻止任何其他线程获取 lock 并 ...

  6. RocketMQ的顺序消费和事务消费

    一.三种消费 :1.普通消费 2. 顺序消费 3.事务消费 1.1  顺序消费:在网购的时候,我们需要下单,那么下单需要假如有三个顺序,第一.创建订单 ,第二:订单付款,第三:订单完成.也就是这个三个 ...

  7. centos上tcp抓包

    tcpdump host 10.1.131.75 -i eth0  -w data.cap 其中,10.1.131.75上目标机器的IP,eth0上网卡名称,data.cap上抓包数据写入的文件.

  8. Vue动态修改网页标题

    业务需求,进入页面的时候,网页有个默认标题,加载的网页内容不同时,标题需要变更. 例:功能授权,功能授权(张三). Vue下有很多的方式去修改网页标题,这里总结下解决此问题的几种方案: 一.最笨方案 ...

  9. Linux搭建www,mail,ftp三大DNS服务器

    ##############################-----服务器端----###############################1. 安装bind# yum install bin ...

  10. 【转载】Visual Studio2017如何设置打包发布的WinForm应用程序的版本号

    在Visual Studio 2017集成开发工具中,打包发布Winform窗体应用程序的时候,支持设置此次打包发布的Winform窗体应用程序对应的版本号信息,并且支持一次设置后,后续的所有发布版本 ...