线程的概念

现在的操作系统几乎都支持运行多个任务,而在操作系统内部,一个任务往往代表的执行的某一个程序,也就是运行中的程序,运行的程序是一个动态的概念,也就是所说的进程,而在进程内部,往往有许多顺序执行流,这些顺序执行流就是线程。

线程的创建

Python提供了 _thread 和 threading 两个模块来支持多线程,其中 _thread 提供低级别的、原始的线程支持,以及一个简单的锁,正如它的名字所暗示的,一般编程不建议使用 thread 模块;而 threading 模块则提供了功能丰富的多线程支持。

Python 主要通过两种方式来创建线程:

  1. 使用 threading 模块的 Thread 类的构造器创建线程。
  2. 继承 threading 模块的 Thread 类创建线程类。

使用Thread类创建线程

Thread类的构造或者说初始化的__init__方法如下:

  1. __init__(self, group=None, target=None, name=None, args=(), kwargs=None, *,daemon=None)

上面的构造器涉及如下几个参数:

  • group:指定该线程所属的线程组。目前该参数还未实现,因此它只能设为 None。
  • target:指定该线程要调度的目标方法。
  • args:指定一个元组,以位置参数的形式为 target 指定的函数传入参数。元组的第一个元素传给 target 函数的第一个参数,元组的第二个元素传给 target 函数的第二个参数……依此类推。
  • kwargs:指定一个字典,以关键字参数的形式为 target 指定的函数传入参数。
  • daemon:指定所构建的线程是否为后代线程。

通过 Thread 类的构造器创建井启动多线程的步骤如下:

  1. 调用 Thread 类的构造器创建线程对象。在创建线程对象时,target 参数指定的函数将作为线程执行体。
  2. 调用线程对象的 start() 方法启动该线程。

示例:

  1. import threading
  2.  
  3. #定义一个函数,该函数返回当前执行的进程名字
  4. def fc(value):
  5. for i in range(value):
        # 调用threading模块current_thread()函数获取当前线程
        # 线程对象的getName()方法获取当前线程的名字
  6. print(threading.current_thread().getName() + "--->" + str(i))
  7.  
  8. #定义线程执行体
  9. for i in range():
  10. print(threading.current_thread().getName() + '--->' + str(i))
  11. if i == : #当if条件成立时,执行子线程
  12. sd = threading.Thread(target=fc, args=(, ))
  13. sd.start()
  14. print("主线程执行完毕")

程序可以通过 setName(name) 方法为线程设置名字,也可以通过 geName() 方法返回指定线程的名字,这两个方法可通过 name 属性来代替。在默认情况下,主线程的名字为 MainThread,用户启动的多个线程的名字依次为 Thread-1、Thread-2、Thread-3、...、Thread-n 等。

继承 threading 模块的 Thread 类创建线程类

通过继承 Thread 类来创建并启动线程的步骤如下:

  1. 定义 Thread 类的子类,并重写该类的 run() 方法。run() 方法的方法体就代表了线程需要完成的任务,因此把 run() 方法称为线程执行体。
  2. 创建 Thread 子类的实例,即创建线程对象。
  3. 调用线程对象的 start() 方法来启动线程。

示例:

  1. import threading
  2.  
  3. class DefineThread(threading.Thread):
  4.  
  5. def __init__(self):
  6. #调用父类的初始化方法。
  7. threading.Thread.__init__(self) #或者
  8. #super().__init__()
  9. self.i =
  10. #重新定义run方法,也就是线程的执行体
  11. def run(self):
  12. while self.i < :
  13. print(threading.current_thread().name + "--->" + str(self.i))
  14. self.i +=
  15.  
  16. #主程序
  17. for i in range():
  18. print(threading.current_thread().name + '--->' + str(i))
  19. if i == :
  20. dt = DefineThread()
  21. dt.start()
  22.  
  23. print("主线程执行完毕")

方法

join方法

Thread 提供了让一个线程等待另一个线程完成的 join() 方法。当调用程序调用另外一个线程的join()方法时,需要等待另外一个线程执行完成才继续执行当前的程序。也就是说当前的执行程序会被阻塞。

示例:

  1. import threading
  2.  
  3. def ft(value):
  4. for i in range(value):
  5. print(threading.current_thread().name + "--->" +str(i))
  6.  
  7. #主程序
  8. for i in range():
  9. if i == :
  10. dt = threading.Thread(target=ft, args=(, ), name = "被其他线程调用join()方法的线程")
  11. dt.start()
  12. dt.join()
  13. print(threading.current_thread().name + "--->" + str(i))
  14.  
  15. print("主线程执行完毕")

输出如下:

  1. MainThread--->
  2. MainThread--->
  3. 被其他线程调用join()方法的线程--->
  4. 被其他线程调用join()方法的线程--->
  5. 被其他线程调用join()方法的线程--->
  6. 被其他线程调用join()方法的线程--->
  7. 被其他线程调用join()方法的线程--->
  8. MainThread--->
  9. MainThread--->
  10. MainThread--->
  11. 主线程执行完毕

可以看出,当子线程在执行的时候,主线程是被阻塞的。

daemon(守护或者后台进程)

daemon属性用来设置线程运行在后台,默认是运行在前台。当所有的前台进程都死掉时,后台进程也结束,不管后台进程是否运行完毕。

示例

  1. import threading
  2.  
  3. def ft(value):
  4. for i in range(value):
  5. print(threading.current_thread().name + "--->" +str(i))
  6.  
  7. t = threading.Thread(target=ft, args=(, ), name = "后台线程", daemon=True)
  8. t.start()
  9. #如果没有在创建线程对象指定daemon=True,那么也可以通过下面的属性设置
  10. #t = threading.Thread(target=ft, args=(, ), name = "后台线程")
  11. #t.daemon = True
  12.  
  13. #主程序
  14. for i in range():
  15. print(threading.current_thread().name + "--->" + str(i))
  16.  
  17. print("主线程执行完毕")

上述程序在执行的时候,正常应该是t这个线程对象执行到99的时候才推出,但是,在主程序内部的线程执行完毕(也就是前台进程完毕),那么后台进程也会随之退出。

从上面的程序可以看出,主线程默认是前台线程,t线程默认也是前台线程。但并不是所有的线程默认都是前台线程,有些线程默认就是后台线程。前台线程创建的子线程默认是前台线程,后台线程创建的子线程默认是后台线程。

可见,创建后台线程有两种方式:

  1. 主动将线程的 daemon 属性设置为 True。
  2. 后台线程启动的线程默认是后台线程。

注意,当前台线程死亡后,Python 解释器会通知后台线程死亡,但是从它接收指令到做出响应需要一定的时间。如果要将某个线程设置为后台线程,则必须在该线程启动之前进行设置。也就是说,将 daemon 属性设为 True,必须在 start() 方法调用之前进行,否则会引发 RuntimeError 异常。

python3之线程(一)的更多相关文章

  1. python3之线程与进程

    1.CPU运行原理 我们都知道CPU的根本任务就是执行指令,对计算机来说最终都是一串由“0”和“1”组成的序列.CPU从逻辑上可以划分成3个模块,分别是控制单元.运算单元和存储单元,这三部分由CPU内 ...

  2. Python3 进程 线程 同步锁 线程死锁和递归锁

    进程是最小的资源单位,线程是最小的执行单位 一.进程 进程:就是一个程序在一个数据集上的一次动态执行过程. 进程由三部分组成: 1.程序:我们编写的程序用来描述进程要完成哪些功能以及如何完成 2.数据 ...

  3. Python3使用线程

    Python2标准库中提供了两个模块thread和threading支持多线程.thread有一些缺陷在Python3中弃用,为了兼容性,python3 将 thread 重命名为 "_th ...

  4. python3线程启动与停止

    转自: https://blog.csdn.net/weixin_38125866/article/details/76795462 https://www.cnblogs.com/lcchuguo/ ...

  5. 洗礼灵魂,修炼python(87)-- 知识拾遗篇 —— 线程(1)

    线程(上) 1.线程含义:一段指令集,也就是一个执行某个程序的代码.不管你执行的是什么,代码量少与多,都会重新翻译为一段指令集.可以理解为轻量级进程 比如,ipconfig,或者, python   ...

  6. python3之协程

    1.协程的概念 协程,又称微线程,纤程.英文名Coroutine. 线程是系统级别的它们由操作系统调度,而协程则是程序级别的由程序根据需要自己调度.在一个线程中会有很多函数,我们把这些函数称为子程序, ...

  7. Python 3 线程模型,进程模型记录

    最近需要使用 python3 多线程处理大型数据,顺道探究了一下,python3 的线程模型的情况,下面进行简要记录: 多线程运行的优点: 使用线程可以把程序中占用时间较长的任务放到后台去处理: 用户 ...

  8. Python之路PythonThread,第四篇,进程4

    python3  进程/线程4 进程间同步互斥方法: from multiprocessing import Lock 创建 进程锁对象 lock = Lock() lock.acquire()  给 ...

  9. Python3 与 C# 并发编程之~ 线程篇

      2.线程篇¶ 在线预览:https://github.lesschina.com/python/base/concurrency/3.并发编程-线程篇.html 示例代码:https://gith ...

随机推荐

  1. bat计算指定文件MD5并输出txt

    @echo off set Name1=*.ADS set Name2=GM_RSSPI* set Name3=equipment* set Name4=protocols* REM 设置输出文件名 ...

  2. 【阿里云开发】- 安装tomcat

    Tomcat配置过程 1.下载Tomcat 官网地址:http://tomcat.apache.org/ 这里我用的是apache-tomcat-8.5.38.tar.gz 2.通过ftp工具将下载好 ...

  3. Python简单的CTF题目hash碰撞小脚本

    Python简单的CTF题目hash碰撞小脚本 import hashlib for num in range(10000,9999999999): res = hashlib.sha1(str(nu ...

  4. gitlab异地备份并验证MD5值

    最近公司发生了蛮多事情的,唉,咱也不管问啊,好好干活吧 需求 把gitlab的备份文件异地备份一份,备份无论失败还是成功通知某邮箱 实现思路 先rsync文件过去,判断rsync这个步骤有没有成功,失 ...

  5. 如何在backoffice里创建Hybris image container以及分配给product

    登录backoffice,在media container视图点击新建按钮: Catalog选择Product Catalog: 在Properties界面,可以选择media实例放入该contain ...

  6. Ubuntu 用户权限相关命令

    目标 用户 和 权限 的基本概念 用户管理 终端命令 组管理 终端命令 修改权限 终端命令 01. 用户 和 权限 的基本概念 1.1 基本概念 用户 是 Linux 系统工作中重要的一环,用户管理包 ...

  7. 输入web网站点击之后发生的事情

    (1)用户在浏览器(客户端)里输入或者点击一个连接: (2)浏览器向服务器发送HTTP请求: (3)服务器处理请求,如果查询字符串或者请求体里含有参数,服务器也会把这些参数信息考虑进去: (4)服务器 ...

  8. systemd_journal_no_entries问题解决

    问题: #journalctl Journal file /var/log/journal/410/system@0.journal~ uses an unsupported feature, ign ...

  9. Flink原理(五)——容错机制

    本文是博主阅读Flink官方文档以及<Flink基础教程>后结合自己理解所写,若有表达有误的地方欢迎大伙留言指出. 1.  前言 流式计算分为有状态和无状态两种情况,所谓状态就是计算过程中 ...

  10. 文件系统属性chattr权限

    命令格式 chattr [+-=] [选项] 文件名或目录名 + 增加权限 - 删除权限 = 等于某权限 i 如果对文件赋予i权限,那么不允许对文件进行删除.改名,也不能添加.修改数据:如果对目录添加 ...