Python 多线程之threading 模块的使用


Posted in Python onApril 14, 2021

简介

Python 通过 _thread 和 threading 模块提供了对多线程的支持,threading 模块兼具了 _thread 模块的现有功能,又扩展了一些新的功能,具有十分丰富的线程操作功能

创建线程

使用 threading 模块创建线程通常有两种方式:

1)使用 threading 模块中 Thread 类的构造器创建线程,即直接对类 threading.Thread 进行实例化,并调用实例化对象的 start 方法创建线程;

2)继承 threading 模块中的 Thread 类创建线程类,即用 threading.Thread 派生出一个新的子类,将新建类实例化,并调用其 start 方法创建线程。

构造器方式

调用 threading.Thread 类的如下构造器创建线程:

threading.Thread(group=None, target=None, name=None, args=(), kwargs={}, *, daemon=None)

group:指定该线程所属的线程组,目前该参数还未实现,为了日后扩展 ThreadGroup 类实现而保留。
target:用于 run() 方法调用的可调用对象,默认是 None,表示不需要调用任何方法。
args:是用于调用目标函数的参数元组,默认是 ()。
kwargs:是用于调用目标函数的关键字参数字典,默认是 {}。
daemon:如果 daemon 不是 None,线程将被显式的设置为守护模式,不管该线程是否是守护模式,如果是 None (默认值),线程将继承当前线程的守护模式属性。
import time
import threading

def work(num):
    print('线程名称:',threading.current_thread().getName(),'参数:',num,'开始时间:',time.strftime('%Y-%m-%d %H:%M:%S'))

if __name__ == '__main__':
    print('主线程开始时间:',time.strftime('%Y-%m-%d %H:%M:%S'))
    
    t1 = threading.Thread(target=work,args=(3,))
    t2 = threading.Thread(target=work,args=(2,))
    t3 = threading.Thread(target=work,args=(1,))
    
    t1.start()
    t2.start()
    t3.start()
    
    t1.join()
    t2.join()
    t3.join()
    
    print('主线程结束时间:', time.strftime('%Y-%m-%d %H:%M:%S'))

上述示例中实例化了三个 Thread 类的实例,并向任务函数传递不同的参数,start 方法开启线程,join 方法阻塞主线程,等待当前线程运行结束。

继承方式

通过继承的方式创建线程包括如下步骤:1)定义 Thread 类的子类,并重写该类的 run 方法;2)创建 Thread 子类的实例,即创建线程对象;3)调用线程对象的 start 方法来启动线程。示例如下:

import time
import threading

class MyThread(threading.Thread):
    
    def __init__(self,num):
        super().__init__()
        self.num = num
    
    def run(self):
        print('线程名称:', threading.current_thread().getName(), '参数:', self.num, '开始时间:', time.strftime('%Y-%m-%d %H:%M:%S'))

if __name__ == '__main__':
   
     print('主线程开始时间:',time.strftime('%Y-%m-%d %H:%M:%S'))
    
    t1 = MyThread(3)
    t2 = MyThread(2)
    t3 = MyThread(1)
   
     t1.start()
    t2.start()
    t3.start()
    
    t1.join()
    t2.join()
    t3.join()
    
    print('主线程结束时间:', time.strftime('%Y-%m-%d %H:%M:%S'))

上述示例中自定义了线程类 MyThread,继承了 threading.Thread,并重写了 __init__ 方法和 run 方法。

守护线程

守护线程(也称后台线程)是在后台运行的,它的任务是为其他线程提供服务,如 Python 解释器的垃圾回收线程就是守护线程。如果所有的前台线程都死亡了,守护线程也会自动死亡。来看个例子:

# 不设置守护线程
import threading

def work(num):
    for i in range(num):
        print(threading.current_thread().name + "  " + str(i))

t = threading.Thread(target=work, args=(10,), name='守护线程')
t.start()

for i in range(10):
    pass
# 设置守护线程
import threading

def work(num):
    for i in range(num):
        print(threading.current_thread().name + "  " + str(i))

t = threading.Thread(target=work, args=(10,), name='守护线程')
t.daemon = True
t.start()

for i in range(10):
    pass

上述示例直观的说明了当前台线程结束,守护线程也会自动结束。

如果你设置一个线程为守护线程,就表示这个线程是不重要的,在进程退出的时候,不用等待这个线程退出;如果你的主线程在退出的时候,不用等待哪些子线程完成,那就设置这些线程为守护线程;如果你想等待子线程完成后再退出,那就什么都不用做,或者显示地将  daemon 属性设置为 false。

线程本地数据

Python 的 threading 模块提供了 local 方法,该方法返回得到一个全局对象,不同线程使用这个对象存储的数据,其它线程是不可见的(本质上就是不同的线程使用这个对象时为其创建一个独立的字典)。来看个示例:

# 不使用 threading.local
import threading
import time

num = 0

def work():
    global num
    
    for i in range(10):
        num += 1
        
    print(threading.current_thread().getName(), num)
    time.sleep(0.0001)
    
for i in range(5):
    threading.Thread(target=work).start()

上面示例中 num 是全局变量,变成了公共资源,通过输出结果,我们发现子线程之间的计算结果出现了互相干扰的情况。

# 使用 threading.local
num = threading.local()

def work():
    num.x = 0
    
    for i in range(10):
        num.x += 1
    
    print(threading.current_thread().getName(), num.x)
    time.sleep(0.0001)

for i in range(5):
    threading.Thread(target=work).start()

使用 threading.local 的示例中,num 是全局变量,但每个线程定义的属性 num.x 是各自线程独有的,其它线程是不可见的,因此每个线程的计算结果未出现相互干扰的情况。

定时器

threading 模块提供了 Timer 类实现定时器功能,来看个例子:

# 单次执行
from threading import Timer

def work():
    print("Hello Python")
    
# 5 秒后执行 work 方法
t = Timer(5, work)
t.start()

Timer 只能控制函数在指定的时间内执行一次,如果我们需要多次重复执行,需要再进行一次调度,想要取消调度时可以使用 Timer 的 cancel 方法。来看个例子:

# 重复执行
count = 0

def work():
    print('当前时间:', time.strftime('%Y-%m-%d %H:%M:%S'))
    global t, count
    count += 1
    # 如果 count 小于 5,开始下一次调度
    if count < 5:
        t = Timer(1, work)
        t.start()

# 指定 2 秒后执行 work 方法
t = Timer(2, work)
t.start()

以上就是Python 多线程之threading 模块的使用的详细内容,更多关于python threading的使用的资料请关注三水点靠木其它相关文章!

Python 相关文章推荐
python实现爬虫下载美女图片
Jul 14 Python
Python实现二维有序数组查找的方法
Apr 27 Python
用Python将动态GIF图片倒放播放的方法
Nov 02 Python
Python 记录日志的灵活性和可配置性介绍
Feb 27 Python
Python实现修改文件内容的方法分析
Mar 25 Python
Python多继承顺序实例分析
May 26 Python
Django 开发调试工具 Django-debug-toolbar使用详解
Jul 23 Python
Python学习笔记之列表和成员运算符及列表相关方法详解
Aug 22 Python
postman传递当前时间戳实例详解
Sep 14 Python
解决TensorFlow训练内存不断增长,进程被杀死问题
Feb 05 Python
python爬虫数据保存到mongoDB的实例方法
Jul 28 Python
python爬虫爬取某网站视频的示例代码
Feb 20 Python
教你如何用python开发一款数字推盘小游戏
深度学习详解之初试机器学习
正确的理解和使用Django信号(Signals)
Apr 14 #Python
编写python程序的90条建议
Apr 14 #Python
Python基础知识之变量的详解
理解深度学习之深度学习简介
Apr 14 #Python
python基于scrapy爬取京东笔记本电脑数据并进行简单处理和分析
You might like
PHP根据IP判断地区名信息的示例代码
2014/03/03 PHP
php使用escapeshellarg时中文被过滤的解决方法
2016/07/10 PHP
php文件包含目录配置open_basedir的使用与性能详解
2017/04/03 PHP
PHP正则+Snoopy抓取框架实现的抓取淘宝店信誉功能实例
2017/05/17 PHP
Nigma vs Liquid BO3 第二场2.13
2021/03/10 DOTA
URI、URL和URN之间的区别与联系
2006/12/20 Javascript
js 操作css实现代码
2009/06/11 Javascript
让div层随鼠标移动的实现代码 ie ff
2009/12/18 Javascript
在JavaScript中获取请求的URL参数
2010/12/22 Javascript
文件编码导致jquery失效的解决方法
2013/06/26 Javascript
JavaScript中连接操作Oracle数据库实例
2015/04/02 Javascript
js获取url传值的方法
2015/12/18 Javascript
JavaScript实现输入框与清空按钮联动效果
2016/09/09 Javascript
关于Javascript中defer和async的区别总结
2016/09/20 Javascript
jquery.multiselect多选下拉框实现代码
2016/11/11 Javascript
JavaScript给每一个li节点绑定点击事件的实现方法
2016/12/01 Javascript
基于JS实现翻书效果的页面切换样式
2017/02/16 Javascript
关于jQuery中fade(),show()起始位置的一点小发现
2017/04/25 jQuery
ES6入门教程之let和const命令详解
2017/05/17 Javascript
浅谈JS获取元素的N种方法及其动静态讨论
2017/08/25 Javascript
vue源码解析之事件机制原理
2018/04/21 Javascript
JS获取浏览器地址栏的多个参数值的任意值实例代码
2018/07/24 Javascript
vue2中,根据list的id进入对应的详情页并修改title方法
2018/08/24 Javascript
在pycharm中开发vue的方法步骤
2020/03/04 Javascript
对python使用http、https代理的实例讲解
2018/05/07 Python
python 获取当天每个准点时间戳的实例
2018/05/22 Python
Python实现基于POS算法的区块链
2018/08/07 Python
解决sublime+python3无法输出中文的问题
2018/12/12 Python
python 字典有序并写入json文件过程解析
2019/09/30 Python
在OpenCV里实现条码区域识别的方法示例
2019/12/04 Python
canvas实现有递增动画的环形进度条的实现方法
2019/07/10 HTML / CSS
Expedia韩国官网:亚洲发展最快的在线旅游门户网站
2018/02/26 全球购物
命名空间(namespace)和程序集(Assembly)有什么区别
2015/09/25 面试题
过程装备与控制工程专业求职信
2014/07/02 职场文书
市场营销毕业求职信
2014/08/07 职场文书
2015初中教导处工作总结
2015/07/21 职场文书