线程池和进程池

一、池的概念

  • 池是用来保证计算机硬件安全的情况下最大限度的利用计算机
  • 它降低了程序的运行效率但是保证了计算机硬件的安全从而让你写的程序能够正常运行
'''
无论是开设进程也好还是开设线程也好 是不是都需要消耗资源
只不过开设线程的消耗比开设进程的稍微小一点而已 我们是不可能做到无限制的开设进程和线程的 因为计算机硬件的资源更不上!!!
硬件的开发速度远远赶不上软件 我们的宗旨应该是在保证计算机硬件能够正常工作的情况下最大限度的利用它
'''

二、线程池

基本使用方式:

from concurrent.futures import ThreadPoolExecutor
import time # 括号内可以传数字,默认会开启cpu个数5倍的线程,传的数字是多少,线程池里面就有几个线程
pool = ThreadPoolExecutor(4) def task(n):
print(n)
time.sleep(2)
return n * 2 # pool.submit(task, 1) #朝池子中提交任务,异步提交
# print('主') t_list = [] #不合理
for i in range(10):
res = pool.submit(task, i) # res -----> <Future at 0x16b06908cc8 state=pending>
#print(res.result()) # 同步提交
t_list.append(res) pool.shutdown()
for i in t_list:
print('>>>>>>>>>>',i.result())
'''
现在是submit一次就得等待一次
为了这5个线程同时启动,可以定义个列表,把线程append到t_list中
然后遍历t_list,再result
等待线程池中的任务执行完毕之后再往下执行:pool.shutdown,关闭线程池 等待线程池中所有的任务运行完毕
''' '''
调用了result后:
程序有并发变成了串行
那么任务的为什么打印的是None?
res.result() 拿到的就是异步提交的任务(task函数)的返回结果,当我们给task函数return一个返回值时,node变成函数task的返回值
'''

三、进程池

import os
from concurrent.futures import ProcessPoolExecutor
import time # 括号内可以传数字,默认会开启你cpu有几个核就开几个,不会超标
pool = ProcessPoolExecutor(5) def task(n):
print(n, os.getpid())
time.sleep(2)
return n * 2 # pool.submit(task, 1) #朝池子中提交任务,异步提交
# print('主') def call_back(n):
#print(n)
print('call_back>>>>>>>>>>>:',n.result()) if __name__ == '__main__':
t_list = []
for i in range(10):
# res = pool.submit(task, i) # res -----> <Future at 0x16b06908cc8 state=pending>
res = pool.submit(task, i).add_done_callback(call_back) #<Future at 0x1b2a6a325c8 state=finished returned int>
# print(res.result()) # 同步提交
#t_list.append(res) '''
现在是submit一次就得等待一次
为了这5个线程同时启动,可以定义个列表,把线程append到t_list中
然后遍历t_list,再result
等待线程池中的任务执行完毕之后再往下执行:pool.shutdown,关闭线程池 等待线程池中所有的任务运行完毕
'''
# pool.shutdown()
# for i in t_list:
# print('>>>>>>>>>>', i.result())
'''
现在是submit一次就得等待一次
为了这5个线程同时启动,可以定义个列表,把线程append到t_list中
然后遍历t_list,再result
等待线程池中的任务执行完毕之后再往下执行:pool.shutdown,关闭线程池 等待线程池中所有的任务运行完毕
''' '''
调用了result后:
程序有并发变成了串行
那么任务的为什么打印的是None?
res.result() 拿到的就是异步提交的任务(task函数)的返回结果,当我们给task函数return一个返回值时,node变成函数task的返回值
'''

总结

from concurrent.futures import ThreadPoolExecutor, ProcessPoolExecutor
pool = ProcessPoolExecutor(5)
pool.submit(task, i).add_done_callback(call_back)

四、协程

'''
进程:资源单位
线程:执行单位
协程:这个概念完全是程序员自己意淫出来的 根本不存在
单线程下实现并发
我们程序员自己再代码层面上检测我们所有的IO操作
一旦遇到IO了 我们在代码级别完成切换
这样给CPU的感觉是你这个程序一直在运行 没有IO
从而提升程序的运行效率
'''

五、gevent模块(了解)

安装

pip3 install gevent -i http://mirrors.aliyun.com/pypi/simple/
from gevent import monkey;monkey.patch_all()
import time
from gevent import spawn """
gevent模块本身无法检测常见的一些io操作
在使用的时候需要你额外的导入一句话
from gevent import monkey
monkey.patch_all()
又由于上面的两句话在使用gevent模块的时候是肯定要导入的
所以还支持简写
from gevent import monkey;monkey.patch_all()
""" def heng():
print('哼')
time.sleep(2)
print('哼') def ha():
print('哈')
time.sleep(3)
print('哈') def heiheihei():
print('heiheihei')
time.sleep(5)
print('heiheihei') start_time = time.time()
g1 = spawn(heng) #把监管的函数名放进去就行,然后得join
g2 = spawn(ha)
g3 = spawn(heiheihei)
g1.join()
g2.join() # 等待被检测的任务执行完毕 再往后继续执行
g3.join()
# heng()
# ha()
# print(time.time() - start_time) # 5.005702018737793
print(time.time() - start_time) # 3.004199981689453 5.005439043045044

六、协程实现TCP服务端的并发

# 服务端
from gevent import monkey;monkey.patch_all()
import socket
from gevent import spawn def communication(conn):
while True:
try:
data = conn.recv(1024)
if len(data) == 0: break
conn.send(data.upper())
except ConnectionResetError as e:
print(e)
break
conn.close() def server(ip, port):
server = socket.socket()
server.bind((ip, port))
server.listen(5)
while True:
conn, addr = server.accept()
spawn(communication, conn) if __name__ == '__main__':
g1 = spawn(server, '127.0.0.1', 8080)
g1.join() # 客户端
from threading import Thread, current_thread
import socket def x_client():
client = socket.socket()
client.connect(('127.0.0.1',8080))
n = 0
while True:
msg = '%s say hello %s'%(current_thread().name,n)
n += 1
client.send(msg.encode('utf-8'))
data = client.recv(1024)
print(data.decode('utf-8')) if __name__ == '__main__':
for i in range(500):
t = Thread(target=x_client)
t.start()

总结

"""
理想状态:
我们可以通过
多进程下面开设多线程
多线程下面再开设协程序
从而使我们的程序执行效率提升
"""

python之线程池和进程池的更多相关文章

  1. Python并发复习4- concurrent.futures模块(线程池和进程池)

    Python标准库为我们提供了threading(多线程模块)和multiprocessing(多进程模块).从Python3.2开始,标准库为我们提供了concurrent.futures模块,它提 ...

  2. Python线程池与进程池

    Python线程池与进程池 前言 前面我们已经将线程并发编程与进程并行编程全部摸了个透,其实我第一次学习他们的时候感觉非常困难甚至是吃力.因为概念实在是太多了,各种锁,数据共享同步,各种方法等等让人十 ...

  3. 协程与concurent.furtrue实现线程池与进程池

    1concurent.furtrue实现线程池与进程池 2协程 1concurent.furtrue实现线程池与进程池 实现进程池 #进程池 from concurrent.futures impor ...

  4. 并发编程:GIL,线程池,进程池,阻塞,非阻塞,同步,异步

    一  GIL(global interpreter lock) GIL中文叫全局解释器锁,我们执行一个文件会产生一个进程,那么我们知道进程不是真正的执行单位,而是资源单位,所以进程中放有解释器(cpy ...

  5. 多进程 multiprocessing 多线程Threading 线程池和进程池concurrent.futures

    multiprocessing.procsess 定义一个函数 def func():pass 在if __name__=="__main__":中实例化 p = process( ...

  6. concurrent.futures模块简单介绍(线程池,进程池)

    一.基类Executor Executor类是ThreadPoolExecutor 和ProcessPoolExecutor 的基类.它为我们提供了如下方法: submit(fn, *args, ** ...

  7. GIL全局解释器锁,线程池与进程池 同步异步,阻塞与非阻塞,异步回调

    GIL全局解释器锁 1.什么是GIL 官方解释:'''In CPython, the global interpreter lock, or GIL, is a mutex that prevents ...

  8. 使用concurrent.futures模块中的线程池与进程池

    使用concurrent.futures模块中的线程池与进程池 线程池与进程池 以线程池举例,系统使用多线程方式运行时,会产生大量的线程创建与销毁,创建与销毁必定会带来一定的消耗,甚至导致系统资源的崩 ...

  9. day 7-7 线程池与进程池

    一. 进程池与线程池 在刚开始学多进程或多线程时,我们迫不及待地基于多进程或多线程实现并发的套接字通信,然而这种实现方式的致命缺陷是:服务的开启的进程数或线程数都会随着并发的客户端数目地增多而增多,这 ...

随机推荐

  1. Prometheus监控神器-服务发现篇(二)

    本章节讲解服务发现与Relabelling的机制与范例. 通过服务发现的方式,我们可以在不重启Prometheus服务的情况下动态的发现需要监控的Target实例信息. 如上图所示,对于线上环境我们可 ...

  2. webapi上传图片的两种方式

    /// <summary>        /// App上传图片        /// </summary>        /// <returns>返回上传图片的 ...

  3. docker 修改容器配置文件

    启动docker镜像命令docker run 可以指定端口映射,但是容器一旦创建就无法在通过命令修改.通常是保存镜像在创建一个新的容器.有没有办法不保存镜像直接修改这个容器呢?答案是有的,本文已mys ...

  4. MyBatis开发重点知识

    1.1为什么需要ORM框架? 传统的JDBC编程存在的弊端: ü 工作量大,操作数据库至少要5步: ü 业务代码和技术代码耦合: ü 连接资源手动关闭,带来了隐患: MyBatis前身是iBatis, ...

  5. 在Linux命令行里与其他用户通信

    大家好,我是良许 在 Linux 命令行里向其他用户发送信息很简单,很多命令都可以做到这点,麻烦的是你需要从众多命令中挑选一个合适的命令来使用.因此,我挑选了四种常用的 Linux 用户通信命令分享给 ...

  6. HTTP走私

    干货 https://paper.seebug.org/1048/

  7. nodejs解压版安装和配置(带有搭建前端项目脚手架)

    nodejs 安装  我先前用了nvm,觉得nvm挺厉害可以随时更换nodejs版本,但是研究了下,可能自己功力不够还是什么,并不好用,中间还出现了错误:所以最后还是卸载了: 本文图文并茂的一步一步的 ...

  8. 我是怎样通过个人项目成长为高级 JavaScript 开发者的

    大道理都是能用三两句话说清的 . 在我开发人员的职业生涯中,最好的决定之一就是不再满足于只做工具的使用者. 我决定创建我自己的工具,并不是为了卖钱,而是要了解这些技术的实际工作方式,并提升我的技术技能 ...

  9. 我的Python自学之路-003 字符串的知识

    '''字符串是以引号或者单引号括起来的任意文本,例如"123","asdfjk",'adfa'引号或者单引号,只是一种表示方法,并不是字符串的一部分如果字符串本 ...

  10. 面试题总结:可能是全网最好的MySQL重要知识点

    标题有点标题党的意思,但希望你在看了文章之后不会有这个想法--这篇文章是作者对之前总结的 MySQL 知识点做了完善后的产物,可以用来回顾MySQL基础知识以及备战MySQL常见面试问题. 什么是My ...