我的编程空间,编程开发者的网络收藏夹
学习永远不晚

怎么理解Python多线程

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

怎么理解Python多线程

本篇内容主要讲解“怎么理解Python多线程”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“怎么理解Python多线程”吧!

在实际处理数据时,因系统内存有限,我们不可能一次把所有数据都导出进行操作,所以需要批量导出依次操作。为了加快运行,我们会采用多线程的方法进行数据处理,以下为我总结的多线程批量处理数据的模板:

import threading# 从数据库提取数据的类class Scheduler():    def __init__(self):        self._lock = threading.RLock()        self.start = 0        # 每次取10000条数据        self.step = 10000    def getdata(self):        # 上锁,以免多线程同时对数据库进行访问,取出重复数据        self._lock.acquire()        # 进行取数据操作        data = 'select * from table' \               'where id between self.start and self.start + self.step'        # 取完数据后,指针后移        self.start += self.step        self._lock.release()        return data# 处理数据的过程写在这里def processdata():    # 从该实例中提取数据    data = scheduler.getdata()    while data:        # 进行处理数据的具体操作:        # 去重、补缺、运算...只要还有数据,本线程就继续取新数据        # 然后再获取数据,进行循环        data = scheduler.getdata()# 创建多线程,threads_num为创建的线程数def threads_scheduler(threads_num):    threads = []    for i in range(threads_num):        # 创建线程        td = threading.Thread(target=processdata, name='th'+str(i+1))        threads.append(td)    for t in threads:        # 启动线程        t.start()        for t in threads:            # 子线程守护            t.join()            print('数据已全部处理成功')if __name__=='__main__':    # 实例化一个调度器,初始化参数    scheduler = Scheduler()    # 创建线程,开始处理数据    threads_scheduler(4)

主要分为三大部分:

  • Scheduler类,负责初始化参数,getdata方法负责提取数据

  • processdata方法中写具体处理数据的流程

  • threads_scheduler方法负责创建线程

Python多线程的知识我分为4部分进行讲解,以下带大家来回顾重点:

多线程threading

本章先为大家介绍了线程的相关概念:

主线程:当一个程序启动时,就有一个进程被操作系统(OS)创建,与此同时一个线程也立刻运行,该线程通常叫做程序的主线程(Main Thread)。因为它是程序开始时就执行的,如果你需要再创建线程,那么创建的线程就是这个主线程的子线程。

子线程:使用threading、ThreadPoolExecutor创建的线性均为子线程。

主线程的重要性体现在两方面:1.是产生其他子线程的线程;2.通常它必须最后完成执行,比如执行各种关闭动作。

在飞车程序中,如果没有多线程,我们就不能一边听歌一边玩飞车,听歌与玩游戏不能并行;在使用多线程后,我们就可以在玩游戏的同时听背景音乐。在这个例子中启动飞车程序就是一个进程,玩游戏和听音乐是两个线程。

Python提供了threading模块来实现多线程:threading.Thread可以创建线程;setDaemon(True)为守护主线程,默认为False;join()为守护子线程。

from time import sleepimport threadingdef music(music_name):    for i in range(2):        print('正在听{}'.format(music_name))        sleep(1)        print('music over')def game(game_name):    for i in range(2):        print('正在玩{}'.format(game_name))        sleep(3)        print('game over')threads = []t1 = threading.Thread(target=music,args=('稻香',))threads.append(t1)t2 = threading.Thread(target=game,args=('飞车',))threads.append(t2)if __name__ == '__main__':    for t in threads:        # t.setDaemon(True)        t.start()            for t in threads:        t.join()    print('主线程运行结束')

线程池

因为新建线程系统需要分配资源、终止线程系统需要回收资源,所以如果可以重用线程,则可以减去新建/终止的开销以提升性能。同时,使用线程池的语法比自己新建线程执行线程更加简洁。

Python为我们提供了ThreadPoolExecutor来实现线程池,此线程池默认子线程守护。它的适应场景为突发性大量请求或需要大量线程完成任务,但实际任务处理时间较短。

from time import sleep# fun为定义的待运行函数with ThreadPoolExecutor(max_workers=5) as executor:    ans = executor.map(fun, [遍历值])    for res in ans:        print(res)with ThreadPoolExecutor(max_workers=5) as executor:    list = [遍历值]    ans = [executor.submit(fun, i) for i in list]    for res in as_completed(ans):        print(res.result())

其中max_workers为线程池中的线程个数,常用的遍历方法有map和submit+as_completed。根据业务场景的不同,若我们需要输出结果按遍历顺序返回,我们就用map方法,若想谁先完成就返回谁,我们就用submit+as_complete方法。

线程互斥

我们把一个时间段内只允许一个线程使用的资源称为临界资源,对临界资源的访问,必须互斥的进行。互斥,也称间接制约关系。线程互斥指当一个线程访问某临界资源时,另一个想要访问该临界资源的线程必须等待。当前访问临界资源的线程访问结束,释放该资源之后,另一个线程才能去访问临界资源。锁的功能就是实现线程互斥。

我把线程互斥比作厕所包间上大号的过程,因为包间里只有一个坑,所以只允许一个人进行大号。当第一个人要上厕所时,会将门上上锁,这时如果第二个人也想大号,那就必须等第一个人上完,将锁解开后才能进行,在这期间第二个人就只能在门外等着。这个过程与代码中使用锁的原理如出一辙,这里的坑就是临界资源。

Python 的 threading 模块引入了锁。threading 模块提供了 Lock 类,它有如下方法加锁和释放锁:

  • acquire():对 Lock加锁,其中timeout参数指定加锁多少秒

  • release():释放锁

class Account:    def __init__(self, card_id, balance):        # 封装账户ID、账户余额的两个变量        self.card_id= card_id        self.balance = balance        def withdraw(account, money):    # 进行加锁    lock.acquire()    # 账户余额大于取钱数目    if account.balance >= money:        # 吐出钞票        print(threading.current_thread().name + "取钱成功!吐出钞票:" + str(money),end=' ')        # 修改余额        account.balance -= money        print("\t余额为: " + str(account.balance))    else:        print(threading.current_thread().name + "取钱失败!余额不足")    # 进行解锁    lock.release()# 创建一个账户,银行卡id为8888,存款1000元acct = Account("8888" , 1000)# 模拟两个对同一个账户取钱# 在主线程中创建一把锁lock = threading.Lock()threading.Thread(name='窗口A', target=withdraw , args=(acct , 800)).start()threading.Thread(name='窗口B', target=withdraw , args=(acct , 800)).start()

lock与Rlock的区别

区别一:Lock被称为原始锁,一个线程只能请求一次;RLock被称为重入锁,可以被一个线程请求多次,即锁中可以嵌套锁。

import threadingdef main():    lock.acquire()    print('第一道锁')    lock.acquire()    print('第二道锁')    lock.release()    lock.release()    if __name__ == '__main__':    lock = threading.Lock()    main()

我们会发现这个程序只会打印“第一道锁”,而且程序既没有终止,也没有继续运行。这是因为Lock锁在同一线程内第一次加锁之后还没有释放时,就进行了第二次acquire请求,导致无法执行release,所以锁永远无法释放,这就是死锁。如果我们使用RLock就能正常运行,不会发生死锁的状态。

区别二:当Lock处于锁定状态时,不属于特定线程,可在另一个线程中进行解锁释放;而RLock只有当前线程才能释放本线程上的锁,不可由其他线程进行释放,所以在使用RLock时,acquire与release必须成对出现,即解铃还须系铃人。

import threadingdef main():    lock.release()    print("在子线程解锁后打印")if __name__ == '__main__':    lock = threading.Lock()    lock.acquire()    t = threading.Thread(target=main)    t.start()

在主线程中定义Lock锁,然后上锁,再创建一个子线程t运行main函数释放锁,结果正常输出,说明主线程上的锁,可由子线程解锁。

如果把上面的锁改为RLock则报错。在实际中设计程序时,我们会将每个功能分别封装成一个函数,每个函数中都可能会有临界区域,所以就需要用到RLock。

import threadingimport timedef fun_1():    print('开始')    time.sleep(1)    lock.acquire()    print("第一道锁")    fun_2()    lock.release()    def fun_2():    lock.acquire()    print("第二道锁")    lock.release()    if __name__ == '__main__':    lock = threading.RLock()    t1 = threading.Thread(target=fun_1)    t2 = threading.Thread(target=fun_1)    t1.start()    t2.start()

一句话总结就是Lock不能套娃,RLock可以套娃;Lock可以由其他线程中的锁进行操作,RLock只能由本线程进行操作。

到此,相信大家对“怎么理解Python多线程”有了更深的了解,不妨来实际操作一番吧!这里是编程网网站,更多相关内容可以进入相关频道进行查询,关注我们,继续学习!

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

怎么理解Python多线程

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

怎么理解Python多线程

本篇内容主要讲解“怎么理解Python多线程”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“怎么理解Python多线程”吧!在实际处理数据时,因系统内存有限,我们不可能一次把所有数据都导出进行操作
2023-06-25

Java多线程怎么理解

本文小编为大家详细介绍“Java多线程怎么理解”,内容详细,步骤清晰,细节处理妥当,希望这篇“Java多线程怎么理解”文章能帮助大家解决疑惑,下面跟着小编的思路慢慢深入,一起来学习新知识吧。1 线程池的优势总体来说,线程池有如下的优势:(1
2023-07-05

C#多线程安全怎么理解

这篇文章主要介绍“C#多线程安全怎么理解”,在日常操作中,相信很多人在C#多线程安全怎么理解问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”C#多线程安全怎么理解”的疑惑有所帮助!接下来,请跟着小编一起来学习吧
2023-06-22

python怎么做多线程

python 提供多线程机制来同时执行多个任务,通过导入 threading 库实现。创建线程涉及指定目标函数(任务代码)并调用 start 方法。同步线程可通过锁、事件或信号实现。结束线程可使用 join 或 terminate 方法。多
python怎么做多线程
2024-05-14

python多线程怎么用

python多线程的使用方法:1、导入threading模块;2、创建线程对象;3、启动线程;4、等待线程执行完毕;5、多个线程同时执行;6、线程同步和通信。Python多线程是指同时使用多个线程执行代码,以实现并发执行任务的目的。Pyth
python多线程怎么用
2023-11-20

怎么理解Python线程安全

这篇文章主要介绍“怎么理解Python线程安全”,在日常操作中,相信很多人在怎么理解Python线程安全问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”怎么理解Python线程安全”的疑惑有所帮助!接下来,请跟
2023-06-16

详解Python多线程

本文实例为大家解析了Python多线程,供大家参考,具体内容如下 1、多线程的理解 多进程和多线程都可以执行多个任务,线程是进程的一部分。线程的特点是线程之间可以共享内存和变量,资源消耗少(不过在Unix环境中,多进程和多线程资源调度消耗差
2022-06-04

Android 多线程处理之多线程详解

handler.post(r)其实这样并不会新起线程,只是执行的runnable里的run()方法,却没有执行start()方法,所以runnable走的还是UI线程。 1.如果像这样,是可以操作ui,但是run还是走在主线程,见打印出来的
2022-06-06

python 多线程处理List

# -*- coding:UTF-8 -*-#"""根据Redis的密码字典,暴力破解"""import redisimport sys,osimport threadingBIN="/usr/local/bin/medusa"#medus
2023-01-31

python多线程怎么应用

本篇内容主要讲解“python多线程怎么应用”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“python多线程怎么应用”吧!一、首先定义一个封装类,主要是保证PyGILState_Ensure,
2023-06-17

python怎么实现多线程

这篇文章将为大家详细讲解有关python怎么实现多线程,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。Python的优点有哪些1、简单易用,与C/C++、Java、C# 等传统语言相比,Python对代码格
2023-06-14

python多线程怎么自定义线程类

这篇文章主要介绍python多线程怎么自定义线程类,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!python的数据类型有哪些?python的数据类型:1. 数字类型,包括int(整型)、long(长整型)和float
2023-06-14

Python多线程怎么使用

本篇内容主要讲解“Python多线程怎么使用”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“Python多线程怎么使用”吧!Python 多线程多线程类似于同时执行多个不同程序,多线程运行有如下优
2023-06-21

python多线程及多线程有序性怎么实现

这篇文章主要介绍了python多线程及多线程有序性怎么实现的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇python多线程及多线程有序性怎么实现文章都会有所收获,下面我们一起来看看吧。前言多线程一般用于同时调用
2023-07-02

python怎么开启多线程

要开启python中的多线程:创建线程对象,指定要执行的任务和参数。启动线程,使其开始执行目标函数。根据需要,使用join()方法等待线程完成。如何在 Python 中开启多线程多线程是 Python 中一种强大的工具,它允许程序同时执行
python怎么开启多线程
2024-05-14

怎么理解并掌握Python线程

这篇文章主要讲解了“怎么理解并掌握Python线程”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“怎么理解并掌握Python线程”吧!1.自定义进程自定义进程类,继承Process类,重写ru
2023-06-25

python多线程死锁现象怎么解决

今天小编给大家分享一下python多线程死锁现象怎么解决的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。一 死锁现象所谓死锁:
2023-07-02

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录