Python 线程的五种状态、守护线程

一、线程的五种状态

大多数线程的生命周期经历有 新建、就绪、运行、阻塞、死亡 这五种状态,对于多线程来说,一个线程不可能始终霸占着CPU,CPU需要在不同线程之间切换,于是线程的状态也会随之改变。

1.新建

当一个线程被创建后,该线程首先会处于新建状态,此时的线程对象并不会表现出任何线程的动态特征,程序也不会执行线程执行体。

2.就绪

当调用线程对象的start()方法后,该线程处于就绪状态,至于何时开始运行完全取决于Python解释器中线程调度器的调度。

不调用start()方法,而仅仅只调用run()方法的话,系统会把该线程当成一个普通对象,run()也只是一个普通方法,在run()方法运行返回之前,其它线程均无法并发执行,也就是说这个线程已经作为普通程序成了主线程的一部分。

3.运行

只有当处于就绪状态的线程获得了CPU,开始执行run()方法后,此时该线程处于运行状态。处于运行状态的线程不可能一直处于运行状态(除非它的线程执行体足够短以至于瞬间就能结束任务),线程调度会使正在运行的线程被中断,目的是使其它线程获得执行的机会,线程调度的细节取决于底层平台采用的策略。这种策略主要有两种:

抢占式调度策略:对于采用抢占式调度策略的系统而言,系统会给每一个可执行线程一小段时间来处理任务,当该时间段过去后,程序会剥夺该线程占用的资源,让其它线程获得执行的机会。在选择下一个线程时,系统会考虑线程的优先级。

协同式调度策略:在协同式调度策略下,只有当线程调用它的sleep()或yield()方法后才会放弃其所占用的资源,也就是说必须由线程主动放弃其所占的资源。

处于运行状态的线程接下来只有两种去路,要么被堵塞,要么由于需要执行的任务处理完毕或异常原因进入死亡。

4.堵塞

线程会因为以下几种原因而被堵塞

  • 线程调用sleep()方法主动放弃它所占用的处理器资源;
  • 线程调用一个阻塞式I/O方法,在该方法返回之前,该线程将被堵塞;
  • 线程试图获得一个锁对象,但该锁对象正被其它线程所持有;
  • 线程在等待某个通知;

线程会因为以下几种情况而解除堵塞

  • 调用的sleep()方法超出了指定时间;
  • 线程调用的阻塞式I/O方法已经返回;
  • 线程成功地获得了试图获取的锁对象;
  • 线程正在等待某个通知时,其它线程发出了通知;

被堵塞的线程解除堵塞后并不会再次进入运行状态而是进入就绪状态,它需要再次等待线程调度器去调度它,只有当它再次获得处理器资源后,它才会进入运行状态。

5.死亡

当线程的任务执行体执行完毕,或者线程执行遇到一些异常情况,该线程将会死亡。不要试图对已死亡的线程再次调用其start()方法,这会导致程序抛出异常。可以用线程对象的is_alive()方法判断线程是否死亡,只有当线程处于新建或者死亡状态时,该方法会返回False。
另外注意,线程一旦被创建,它不会受到主线程的影响,它和主线程具有同等地位,主线程结束,其它线程也不会随之结束。

二、守护线程

之前我们接触的线程都是在前台运行的,它们也叫前台线程,而有的线程在后台运行,它运行的目的是为了给其它线程提供服务,这样的线程称为后台线程,也叫做守护线程。例如,Python解释器中的垃圾回收线程就是一种后台线程。

所有线程都有一个daemon属性。它用来把线程设置为后台线程,因为它默认为False,所以我们此前创建的线程都是前台线程,当把它设置为True时,该线程就成了后台线程。

后台线程依托于前台线程,当前台线程已经全部死亡的时候,后台线程也会随之死亡。

示例:

import threading
from time import sleep
def text(n):
    for i in range(n):
        print('thread %s is running >> %d.' % (threading.current_thread().name,i))
        if i == n-1:
            print('thread %s end.' % (threading.current_thread().name))
        sleep(1)
t = threading.Thread(target=text,args=(10,),name='后台线程')
t.daemon = True
t.start()
for i in range(5):
    print('thread %s is running >> %d' % (threading.current_thread().name,i))

运行结果:

thread 后台线程 is running >> 0.
thread MainThread is running >> 0
thread MainThread is running >> 1
thread MainThread is running >> 2
thread MainThread is running >> 3
thread MainThread is running >> 4

<分析>

可以看到,后台线程只执行了一次循环,之后就被强制结束,因为主线程结束了,所以后台线程也就没有了存在下去的必要,因此随之结束。

另外,创建后台线程并非只有如上设置daemon属性一种方法,除了设置daemon属性外,后台线程启动的线程也默认是后台线程。
此前提到避免使用_thread模块的其中一个原因就在于_thread模块并不支持守护线程的功能。

本篇参考自 :http://c.biancheng.net/view/2610.html

  • 2
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值