深入了解Python的多线程基础
作者:程序员-夏天 发布时间:2021-12-07 18:50:50
线程
线程(Thread
),有时也被称为轻量级进程(Lightweight Process,LWP
),是操作系 * ⽴调度和分派的基本单位,本质上就是一串指令的集合。
⼀个标准的线程由线程id、当前指令指针(PC),寄存器集合和堆栈组成,它是进程中的⼀个实体,线程本身不拥有系统资源,只拥有⼀点⼉在运⾏中必不可少的资源(如程序计数器、寄存器、栈),但它可与同属⼀个进程的其它线程共享进程所拥有的全部资源。线程不能够独⽴执⾏,必须依存在进程中。
多线程
多线程就是使用多个线程同时执行任务,实现了任务的并行执行,从而提高程序运行效率的方法。
试想一下,如果在单个线程内执行多个任务(比如发送网络请求等),如果前面的任务比较耗时,而后面的任务需要等待前面的任务执行完才能执行,这样会影响任务执行效率,那么就可以使用多线程去执行这些任务,任务可以同时进行,那么将大大的提高执行效率。
Python多线程
在Python中,提供了threading模块来实现多进程操作,这个模块是基于较低级的模块 _thread 的基础上建立的,提供了更易用的高级多线程API。
创建线程
可以通过threading模块中的Thread类来创建线程对象。
Thread语法结构:
threading.Thread(group, target, name, daemon)
group
:默认为None(该参数是为了以后实现ThreadGroup类而保留的)target
:在run方法中调用的可调用对象,即线程要执行的任务name
:线程名称,可以不设定,默认为"Thread-N"形式的名称args
:给target指定的函数传递的参数,以元组的⽅式传递kwargs
:给target指定的函数传递命名参数daemon
:默认为None,将显式地设置该线程是否为守护模式。如果是None,线程将继承当前线程的守护模式属性
Thread常用方法
start()
:启动线程,并调用该线程中的run()方法run()
:线程启动时运行的方法,正是它去调用target指定的函数join(timeout=None)
:让当前调用者线程(一般为主线程)等待,直到该线程结束,timeout是可选的超时时间is_alive()
:返回当前线程是否存活
import threading
import time
def work(i):
print("子线程'{}'work正在运行......".format(threading.current_thread().name))
time.sleep(i)
print("子线程'{}'运行结束......".format(threading.current_thread().name))
if __name__ == '__main__':
print("主线程{}启动".format(threading.current_thread().name))
# 获取线程的名称
threads = []
for i in range(5):
t = threading.Thread(target=work, args=(i,))
# 启动线程
threads.append(t)
t.start()
for t in threads:
t.join()
print("主线程结束")
执行结果为:
上述代码中使用t.join()的功能就是让主线程等待所有子线程结束后才结束,如果想设置守护线程(主线程结束,子线程也随之结束,无论任务执行完成与否)的话,可以使用t.daemon = True
。
GIL锁
GIL的全称是Global Interpreter Lock
(全局解释器锁),这个锁最初的设计是为了保证同一份数据不能被多个线程同时修改,每个线程在执行任务的时候都需要先获取GIL,保证同一时刻只有一个线程可以执行,即同一时刻只有一个线程在解释器中运行,因此Python中的多线程是假的多线程,不是真正意义上的多线程。 如果程序中有多个线程执行任务,那么多个线程会被解释器轮流执行,只不过是切换的很快、很频繁,给人一种多线程“同时”在执行的错觉。
线程池
在之前的文章说过,进程有进程池的机制,同样,线程也有线程池。线程池可以在程序启动时就创建自定义数量的空闲的线程,程序只要将一个任务提交给线程池,线程池就会启动一个空闲的线程来执行它。当该任务执行结束后,该线程并不会死亡,而是再次返回到线程池中变成空闲状态,等待下一个任务的执行。
multiprocessing.dummy
里面也有一个Pool对象,它其实就是线程的封装,使用起来和multiprocessing
的Pool非常类似。它们api都是通用的,简单地说,multiprocessing.dummy
是multiprocessing
进程池模块复制的一个线程池模块,强调一下,这里线程池也是受到GIL限制的。
使用方式和multiprocessing.Pool一致,具体参考Python进程池。
from multiprocessing.dummy import Pool
import time
def work(i):
print("work'{}'执行中......".format(i))
time.sleep(2)
print("work'{}'执行完毕......".format(i))
if __name__ == '__main__':
# 创建线程池
# Pool(5) 表示创建容量为5个线程的线程池
pool = Pool(5)
for i in range(10):
pool.apply_async(work, (i, ))
pool.close()
pool.join()
来源:https://blog.csdn.net/weixin_50097774/article/details/121443474


猜你喜欢
- 主要原理:调整dicom的窗宽,使之各个像素点上的灰度值缩放至[0,255]范围内。使用到的python库:SimpleITK下面是一个将d
- 1 栈的概念栈由一系列对象对象组织的一个集合,这些对象的增加和删除操作都遵循一个“后进先出”(Las
- 腾讯“月捐计划”倡导爱心人士,通过每月小额捐款的形式,长期关注和支持公益项目。并和亿万爱心网友一起,每人每月1份爱,点滴付出,汇成爱海,形成
- 在用户登录windows操作系统的时候,如果触发到了登录表单的密码录入框上,并且此时按下了“大写锁定键(Caps Lock)”,那么界面上会
- 本文实例讲述了python实现获取序列中最小的几个元素。分享给大家供大家参考。具体方法如下:import heapq import rand
- 本文实例讲述了Python实现判断字符串中包含某个字符的判断函数。分享给大家供大家参考,具体如下:#coding=utf8#参数包含两个:#
- 首先,我用的mysql版本号是5.5.17的,最近遇到的问题是在eclipse中通过jdbc往数据库写中文数据的时候,在数据库中显示的是??
- 本文实例讲述了mysql存储过程之游标(DECLARE)原理与用法。分享给大家供大家参考,具体如下:我们在处理存储过程中的结果集时,可以使用
- 1、使用说明首先说明,本文所使用的功能为pycharm专业版所支持,其他版本如社区版,教育版,则不一定支持。作为一名后端开发,我猜你的桌面上
- 本文实例讲述了php验证session无效的解决方法。分享给大家供大家参考。具体方法如下:一、问题今天在配置 apache+php环境时折腾
- pyd文件生成安装easycython库pip install easycythontest.pydef test(): pri
- 1、from子句组装来自不同数据源的数据; 2、where子句基于指定的条件对记录行进行筛选; 3、group&nb
- 1.切片# 切片:取list或tuple的部分元素nameList = ["Willard","ChenJD&
- 首先得抛出个重要的前提观点,大屏幕、宽屏幕根本不是为方便“扫视”准备的。所以说以内容为基础的网站设计不用考虑超宽、超大的场景,甚至包括服务类
- 在Python编程中,导入文本文件是常见的操作之一。Python提供了丰富的标准库,使得文件操作变得十分简单。那么,如何在Python中导入
- 要读懂这些代码主要是要了解ASP中操作二进制数据的对象ADODB.Stream!本程序主要用的就是Adodb.Stream,如果你有这个基础
- 问题:用pycharm每次修改代码后第一次运行还是原来的结果,运行第二次的时候才是修改后代码的结果解决:每次修改代码后保存一下即可解决补充:
- 本篇不是教给大家如何去学习python,有需要详细深入学习的朋友可以参阅:Python基础语言学习笔记总结(精华)本文通过一周快速学习pyt
- 在pyhton中,经常会用到input()语句,但是input()语句输入的内容只能时字符串类型,而我们经常要输入int类型的数据等,那么就
- 简介:fixture区别于unnitest的传统单元测试(setup/teardown)有显著改进:1.有独立的命名,并通过声明它们从测试函