守护线程

概念

有一种线程,它是在后台运行的,它的任务是为其他线程提供服务,这种线程被称为“后台线程(Daemon Thread)”,又称为“守护线程”或“精灵线程”。Python 解释器的垃圾回收线程就是典型的后台线程。

特点

后台线程有一个特征,如果所有的前台线程都死亡了,那么后台线程会自动死亡。

tips:

无论是进程还是线程,都遵循:守护xx会等待主xx运行完毕后被销毁。需要强调的是:运行完毕并非终止运行。

  1. 对主进程来说,运行完毕指的是主进程代码运行完毕
  2. 对主线程来说,运行完毕指的是主线程所在的进程内所有非守护线程统统运行完毕,主线程才算运行完毕

详细解释:

  1. 主进程在其代码结束后就已经算运行完毕了(守护进程在此时就被回收),然后主进程会一直等非守护的子进程都运行完毕后回收子进程的资源(否则会产生僵尸进程),才会结束。
  2. 主线程在其他非守护线程运行完毕后才算运行完毕(守护线程在此时就被回收)。因为主线程的结束意味着进程的结束,进程整体的资源都将被回收,而进程必须保证非守护线程都运行完毕后才能结束。

创建方式

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

 

用户线程

概念

用户线程可以认为是系统的工作线程,它会完成这个程序要完成的业务员操作。如果用户线程全部结束,则意味着这个程序无事可做。守护线程要守护的对象已经不存在了,那么整个应用程序就应该结束。

特点

用户线程全部执行完毕,主线程才算执行完毕

创建方式

python 创建的线程默认是用户线程

 

示例1:  主线程结束后,守护线程即结束执行

from threading import Thread
import os,time,random

def task():
    # t=Thread(target=time.sleep,args=(3,))
    # t.start()
    print('%s is running' %os.getpid())
    time.sleep(2)
    print('%s is done' %os.getpid())

if __name__ == '__main__':
    t=Thread(target=task)
    t.daemon = True
    t.start()
    print('主')

》》》?》

13368 is running
主

示例2:存在用户未结束的用户线程,守护线程不会被强制结束

 

from threading import Thread
import time

def foo():
    print(123)
    time.sleep(1)
    print("end123")


def bar():
    print(456)
    time.sleep(3)
    print("end456")


if __name__ == '__main__':
    t1 = Thread(target=foo)
    t2 = Thread(target=bar)

    t1.daemon = True

    t2.start()
    t1.start()
    print("main-------")

>>>>>>>

456
main-------
123
end123
end456

t2为用户线程,主线程main-----打印完后,t2用户线程还在执行,此时主线程没有结束,所以守护线程还在继续执行

join来阻塞执行

import time


def foo():
    print(123)
    time.sleep(1)
    print("end123")


def bar():
    print(456)
    time.sleep(3)
    print("end456")


if __name__ == '__main__':
    t1 = Thread(target=foo)
    t2 = Thread(target=bar)

    t1.daemon = True
    t2.daemon = True
    t2.start()
    t1.start()
    t2.join()
    t1.join()
    print("main-------")

》》》》》   join来阻塞主线程的执行

456
123
end123
end456
main-------