线程和Python

本节主要记录如何在 Python 中使用线程,其中包括全局解释器锁对线程的限制和对应的学习脚本。

全局解释器锁

Python 代码的执行是由 Python 虚拟机(又叫解释器主循环)进行控制的。

  • 对 Python 虚拟机的访问是由全局解释器锁(GIL)控制的。步骤为:

    1. 设置 GIL;

      1. 切换进一个线程去运行;

      2. 执行下面操作之一:

        1. a. 指定数量的字节代码指令;

        2. b. 线程主动让出控制权(可以调用 time.sleep(0) 来完成)。

      3. 把线程设置回睡眠状态(切换出线程);

      4. 解锁 GIL;

      5. 重复上述步骤

退出线程

当一个线程完成函数的执行时,它就会退出。

  • 退出的方法还有:

    • 通过调用如 thread.exit() 之类的退出函数;

    • 如 sys.exit() 之类的退出 Python 进程的标准方法;

    • 抛出 SystemExit 异常,来使线程退出。

  • 不能直接“终止”一个线程;

在 Python 中使用线程

Python 虽然支持多线程编程,但是还需要取决于它所运行的操作系统。

  • 支持多线程的操作系统:绝大多数类 UNIX 平台(如 Linux、Solaris、Mac OS X、*BSD等)以及 Windows 平台;

  • 要确定解释器(或者说你的系统)是否支持线程,可以从交互式解释器中尝试导入 thread 模块,如果可用则不会产生错误。如下:

    • >>> import thread

  • 如果 Python 解释器没有将线程支持编译进去,模块导入将会失败

    • >>> import thread

      Traceback (most recent call last): File "<stdin>", line 1, in <module>ModuleNotFoundError: No module named 'thread'

**注:在 Python 3 中 thread 模块被重命名为 _thread (原因为平常使用不推荐用 thread 模块,只建议那些想访问线程的更底层级别的专家使用)。

不使用线程的情况

通过一个简单的例子代码来演示在不使用线程的情况下是如何工作的,方便后面对比使用线程的情况。以下为代码:

 #!/usr/bin/env python

from time import sleep, ctime

def loop0():
print('start loop 0 at:', ctime())
sleep(4)
print('loop 0 done at:', ctime())

def loop1():
print('start loop 1 at:', ctime())
sleep(3)
print('loop 1 done at:', ctime())

def main():
print('starting at:', ctime())
loop0()
loop1()
print('all DONE at:', ctime())

if __name__ == '__main__':
main()

运行后的输出结果:

 starting at: Sun Jul 22 17:15:13 2018
start loop 0 at: Sun Jul 22 17:15:13 2018
loop 0 done at: Sun Jul 22 17:15:17 2018
start loop 1 at: Sun Jul 22 17:15:17 2018
loop 1 done at: Sun Jul 22 17:15:20 2018
all DONE at: Sun Jul 22 17:15:20 2018

Python 的线程模块

Python 提供了多个模块来支持多线程编程,包括 thread 、threading 和 Queue 模块等。

  • thread 模块提供了基本的线程和锁定支持;

  • threading 模块提供了更高级别、功能更全面的线程管理;

  • Queue 模块可以创建一个队列数据结构,用于在多线程之间进行共享;

  • 在实际进行多线程编程过程中应该避免使用 thread 模块:

    • threading 模块更加先进、有更好的进程支持,thread 模块中的一些熟悉会和 threading 模块冲突;

    • thread 模块拥有的同步原语注释只有一个(同步原语用于控制执行和访问);

      [注释]  常见的比如锁(Lock)、可重入锁(RLock)
    • thread 模块对进程何时退出没有控制。当主线程结束时,所有其它线程也都强制结束,不会发出警告或者进行适当的清理。

线程和Python—Python多线程编程的更多相关文章

  1. Python的多线程编程

    提到多线程,很多人就会望而却步,本文将由浅入深地带你攻克python多线程编程,并防止你跳入深坑, 首先看一段简单的代码: from time import ctime,sleep def play_ ...

  2. python --- 基础多线程编程

    在python中进行多线程编程之前必须了解的问题: 1. 什么是线程? 答:线程是程序中一个单一的顺序控制流程.进程内一个相对独立的.可调度的执行单元,是系统独立调度和分派CPU的基本单位指运行中的程 ...

  3. 【转】使用python进行多线程编程

    1. python对多线程的支持 1)虚拟机层面 Python虚拟机使用GIL(Global Interpreter Lock,全局解释器锁)来互斥线程对共享资源的访问,暂时无法利用多处理器的优势.使 ...

  4. 【Python】多线程编程

    1.thread模块 2.threading模块 3.Queue模块与多线程互斥 简介: thread和threading模块允许创建和管理线程,thread模块提供了基本的线程和锁的支持,而thre ...

  5. Delphi中线程类TThread实现多线程编程1---构造、析构……

    参考:http://www.cnblogs.com/rogee/archive/2010/09/20/1832053.html Delphi中有一个线程类TThread是用来实现多线程编程的,这个绝大 ...

  6. 转发 Delphi中线程类TThread 实现多线程编程

    Delphi中有一个线程类TThread是用来实现多线程编程的,这个绝大多数Delphi书藉都有说到,但基本上都是对TThread类的几个成员作一简单介绍,再说明一下Execute的实现和Synchr ...

  7. Python:多线程编程

    1.IO编程 IO(input/output).凡是用到数据交换的地方,都会涉及io编程,例如磁盘,网络的数据传输.在IO编程中,stream(流)是一种重要的概念,分为输入流(input strea ...

  8. python爬虫多线程编程

    #使用了线程库 import threading from queue import Queue from bs4 import BeautifulSoup import json import re ...

  9. Delphi中线程类TThread实现多线程编程2---事件、临界区、Synchronize、WaitFor……

    接着上文介绍TThread. 现在开始说明 Synchronize和WaitFor 但是在介绍这两个函数之前,需要先介绍另外两个线程同步技术:事件和临界区 事件(Event) 事件(Event)与De ...

  10. python 多线程编程

    这篇文章写的很棒http://blog.csdn.net/bravezhe/article/details/8585437 使用threading模块实现多线程编程一[综述] Python这门解释性语 ...

随机推荐

  1. Java中常用的七个阻塞队列第二篇DelayQueue源码介绍

    Java中常用的七个阻塞队列第二篇DelayQueue源码介绍 通过前面两篇文章,我们对队列有了了解及已经认识了常用阻塞队列中的三个了.本篇我们继续介绍剩下的几个队列. 本文主要内容:通过源码学习De ...

  2. Acid靶机渗透

    Acid渗透靶机实战 攻击机:kali 192.168.41.147 靶机: acid 192.168.41.149 信息收集 ip发现 开启Acid靶机,通过nmap进行局域网存火主机扫描.![]( ...

  3. PHP 将字符串转换为字符集格式UTF8/GB2312/GBK 函数iconv()

     iconv()介绍 iconv函数可以将一种已知的字符集文件转换成另一种已知的字符集文件 iconv('要转化的格式',‘转化后的格式’,‘转化的数据’); 但是转化是经常出错,一般需要在转成的编码 ...

  4. python第三方库安装与卸载

    一.检查python环境是否正常 python安装完毕并设置环境变量后,可在cmd中运行python查看,显示版本等信息  二.查看已经安装的第三方库 通过pip list可查看已安装的库,以及对应的 ...

  5. 形象地展示信号与系统中的一些细节和原理——卷积、复数、傅里叶变换、拉普拉斯变换、零极图唯一确定因果LTI系统

    看懂本文需要读者具备一定的微积分基础.至少开始学信号与系统了本文主要讲解欧拉公式.傅里叶变换的频率轴的负半轴的意义.傅里叶变换的缺陷.为什么因果LTI系统可以被零极图几乎唯一确定等等容易被初学者忽略但 ...

  6. python os模块获取指定目录下的文件列表

    bath_path = r"I:\ner_results\ner_results" dir_list1 = os.listdir(bath_path) for dir1 in di ...

  7. shell脚本:备份数据库、代码上线

    备份MySQL数据库场景:一台MySQL服务器,跑着5个数据库,在没有做主从的情况下,需要对这5个库进行备份 需求:1)每天备份一次,需要备份所有的库2)把备份数据存放到/data/backup/下3 ...

  8. Unity 芯片拼图算法

    很多游戏的养成系统中会有利用芯片或者碎片来合成特定道具的功能,或者来给玩家以额外的属性提升等,先截个图以便更好说明: 如上图,我们有各种各样形状迥异的碎片,上面只不过列举了其中一部分,现在,我们需要利 ...

  9. Java同步方法:synchronized到底锁住了谁?

    目录 前言 同步方法 类的成员方法 类的静态方法 同步代码块 总结 其他同步方法 参考资料 前言 相信不少同学在上完Java课后,对于线程同步部分的实战,都会感到不知其然. 比如上课做实验的时候,按着 ...

  10. PHP(ThinkPHP5.0) + PHPMailer 进行邮箱发送验证码

    GitHub下载最新版第三方类库PHPMailer: 第一步: 打开网址https://github.com/PHPMailer/PHPMailer/ 下载PHPMailer,PHPMailer 需要 ...