python - threading 多线程 setDaemon 和 join 的区别

#! /usr/bin/env python
"""
python 多线程的管理机制
"""
__author__ = 'sallency'

import threading
import time

#封装一个线程包的类
class MyThread(threading.Thread):

    def __init__(self, name, count, interval):
        threading.Thread.__init__(self)
        self.name = name
        self.count = count
        self.interval = interval
        self.stop = False

    def run(self):
        while not self.stop:
            print "thread: %s count %d time %s" % (self.name, self.count, time.ctime())
            time.sleep(self.interval)
            self.count += 1

    def stop(self):
        self.stop = True
#任务
def task():
    thr_1 = MyThread('thread_1', 10, 3)
    thr_2 = MyThread('thread_2', 5, 3)
    #如果这里设为true的话 则主线程执行完毕后会将子线程回收掉
    #默认是 false 则主进程执行结束时不会回收子线程
    thr_1.setDaemon(True)
    thr_2.setDaemon(True)
    thr_1.start()
    thr_2.start()
    #join则是阻塞主线程 让其在子线程执行完毕后方可继续执行
    #这就保证了当主线程执行完毕前,所有的子线程一定执行完毕了
    #thr_1.join()
    #thr_2.join()
    return True

if __name__ == "__main__":
    print "main threading start: %s" % (time.ctime())
    task()
    print "main threading end: %s" % (time.ctime())

python 可以方便的使用 threading 包来实现多线程

线程对象有两个用来管理线程机制的方法: setDaemon 和 join

     主线程启动若干个子线程后,可以继续执行主线程的代码,也可以等待所有的子线程执行完毕后继续执行主线程,这里需要用到的就是 join 方法,子线程通过调用 join 可以告诉主线程,你必须等着我,我完事了你才可以再往下执行。

     这里要理解,比如 子线程1 花费 10秒,子线程2 花费 5秒,如果子线程 2 调用了 join,那么 主线程只会等待用时 5秒 的子线程2 执行完毕,会继续向下执行,而不会等待还需要5秒才能执行完毕的子线程1

    所以如果需要所有的子线程都能在主线程结束前被执行完毕,则必须为每一个子线程都注册 join

    如果没有为子线程注册 join,则可能会出现在主线程执行完毕之前,还有很多子线程没有执行完毕,这时如果你为子线程注册了 setDaemon(True) 的话,主线程会回收此子线程;否则,主线程就不管他了,自己结束了,子线程依旧在那执行。默认是 False,也就说主线程不会回收子线程

setDaemon

    setDaemon() : 设置此线程是否被主线程守护回收。默认False不回收,需要在 start 方法前调用;设为True相当于像主线程中注册守护,主线程结束时会将其一并回收

join

    join(): 设置主线程是否同步阻塞自己来待此线程执行完毕。如果不设置的话则主进程会继续执行自己的,在结束时根据 setDaemon 有无注册为守护模式的子进程,有的话将其回收,没有的话就结束自己,某些子线程可以仍在执行

转载于:https://my.oschina.net/sallency/blog/677129

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 在 Pythonthreading 模块中,setDaemon() 方法是用来设置线程为守护线程的方法。守护线程是一种特殊的线程,它会随着主线程的结束而自动退出,而不管它是否完成了自己的任务。 如果一个线程被设置为守护线程,那么它不会阻止 Python 解释器退出。这对于一些需要在后台运行的任务非常有用,比如日志记录、定时任务等。 setDaemon() 方法的语法如下: threading.Thread.setDaemon(daemonic) 其中,daemonic 参数是一个布尔值,表示线程是否为守护线程。如果 daemonic 为 True,那么线程就被设置为守护线程;否则,线程就是普通线程。 例如,可以使用以下代码将一个线程设置为守护线程: ``` import threading def worker(): while True: print('Worker is running...') time.sleep(1) t = threading.Thread(target=worker) t.setDaemon(True) # 设置为守护线程 t.start() # 主线程结束,守护线程也会自动退出 ``` ### 回答2: 在Python中,threading模块的setDaemon方法用于设置线程的守护属性。守护线程是一种后台线程,它会随着主线程的结束而自动退出,无论守护线程是否执行完毕。而非守护线程则会等待所有守护线程执行完毕才会退出。 当调用线程的setDaemon(True)方法后,该线程就被设置为守护线程。守护线程的特点是,当所有的非守护线程结束后,守护线程会自动退出,不论它是否执行完毕。守护线程的目的通常是为了提供一种服务或者监控的功能,它并不关心它的工作是否完成,只要主线程执行完毕就可以退出。 守护线程的用途可举例如下: - 后台日志记录:当主线程运行时,守护线程可以记录日志信息,不干扰主线程的运行。 - 定时任务监控:守护线程可以周期性地执行某些操作,如备份文件、清理缓存等。 - 后台检测功能:守护线程可以检测某些任务的状态,并在需要时进行相应的处理。 需要注意的是,守护线程不能创建子线程,因为子线程没有被设置为守护线程,所以只有等到主线程执行完毕后才退出。此外,守护线程中的代码不能含有涉及I/O操作的部分,因为程序退出时,这些I/O操作可能会被中断导致数据丢失。 总之,使用setDaemon方法可以将线程设置为守护线程,使其在主线程结束时自动退出。在合适的情况下,守护线程可以提供一些后台服务或监控功能,从而增强多线程程序的灵活性和功能性。 ### 回答3: `setDaemon` 是 Python 中 `threading` 模块中的一个方法,该方法用于设置线程是否为后台线程。 在多线程编程中,有两种类型的线程:前台线程和后台线程。前台线程是主程序创建的线程,它会阻塞主程序的执行直到线程执行完毕。而后台线程是在主程序执行完毕后自动退出的线程。 当一个线程被设置为后台线程时,它会随着主线程的结束而结束,不论该线程是否执行完毕。而如果一个线程为前台线程,则必须等到该线程执行完毕才能退出程序。 语法:`thread.setDaemon(boolean)` `setDaemon` 方法接受一个布尔值参数,当参数为 `True` 时,将该线程设置为后台线程,当参数为 `False` 时,将该线程设置为前台线程。 一般来说,如果主程序只需等待前台线程执行完毕,而对于后台线程的执行无需关心,可以将后台线程设置为后台线程。这样可以确保主程序可以及时退出,而无需等待后台线程的运行完毕。但需要注意的是,设置线程为后台线程后,对于任何未结束的任务,都不会通过 `join` 等待它的结束。 需要注意的是,`setDaemon` 方法必须在 `start` 方法之前调用,否则会引发异常。 下面是一个使用 `setDaemon` 方法的例子: ```python import threading def my_func(): for i in range(3): print("This is a background thread.") thread = threading.Thread(target=my_func) thread.setDaemon(True) # 将线程设置为后台线程 thread.start() print("This is the main thread.") ``` 以上代码中,`my_func` 函数是一个后台线程,它将会在主线程结束后自动退出。主程序中的 `print` 语句会在主线程执行完毕后打印输出。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值