Python中threading的join和setDaemon的区别及用法

Python多线程编程时经常会用到join()和setDaemon()方法,基本用法如下:

join([time]): 等待至线程中止。这阻塞调用线程直至线程的join() 方法被调用中止-正常退出或者抛出未处理的异常-或者是可选的超时发生。

setDaemon,将该线程标记为守护线程或用户线程

1、join ()方法:主线程A中,创建了子线程B,并且在主线程A中调用了B.join(),那么,主线程A会在调用的地方等待,直到子线程B完成操作后,才可以接着往下执行,那么在调用这个线程时可以使用被调用线程的join方法。
原型:join([timeout]),里面的参数时可选的,代表线程运行的最大时间,即如果超过这个时间,不管这个此线程有没有执行完毕都会被回收,然后主线程或函数都会接着执行的。


1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

import threading

import time

class MyThread(threading.Thread):

    def __init__(selfid):

        threading.Thread.__init__(self)

        self.id = id

    def run(self):

        = 0

        time.sleep(10)

        print(self.id)

        print(‘线程结束:‘+str(time.time()))

if __name__ == "__main__":

    t1 = MyThread(999)

    print(‘线程开始:‘+str(time.time()))

    t1.start()

    print(‘主线程打印开始:‘+str(time.time()))

    for in range(5):

        print(i)

    time.sleep(2)

    print(‘主线程打印结束:‘ + str(time.time()))

线程开始:1497534590.2784667
主线程打印开始:1497534590.2794669
0
1
2
3
4
主线程打印结束:1497534592.279581
999
线程结束:1497534600.2800388

从打印结果可知,线程t1 start后,主线程并没有等线程t1运行结束后再执行,而是在线程执行的同时,执行了后面的语句。

现在,把join()方法加到启动线程后面(其他代码不变)


1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

import threading

import time

class MyThread(threading.Thread):

    def __init__(selfid):

        threading.Thread.__init__(self)

        self.id = id

    def run(self):

        = 0

        time.sleep(10)

        print(self.id)

        print(‘线程结束:‘+str(time.time()))

if __name__ == "__main__":

    t1 = MyThread(999)

    print(‘线程开始:‘+str(time.time()))

    t1.start()

    t1.join()

    print(‘主线程打印开始:‘+str(time.time()))

    for in range(5):

        print(i)

    time.sleep(2)

    print(‘主线程打印结束:‘ + str(time.time()))

线程开始:1497535176.5019968
999
线程结束:1497535186.5025687
主线程打印开始:1497535186.5025687
0
1
2
3
4
主线程打印结束:1497535188.5026832

线程t1 start后,主线程停在了join()方法处,等子线程t1结束后,主线程继续执行join后面的语句。

2、setDaemon()方法。主线程A中,创建了子线程B,并且在主线程A中调用了B.setDaemon(),这个的意思是,把主线程A设置为守护线程,这时候,要是主线程A执行结束了,就不管子线程B是否完成,一并和主线程A退出.这就是setDaemon方法的含义,这基本和join是相反的。此外,还有个要特别注意的:必须在start() 方法调用之前设置。


1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

import threading

import time

class MyThread(threading.Thread):

    def __init__(selfid):

        threading.Thread.__init__(self)

        self.id = id

    def run(self):

        = 0

        time.sleep(10)

        print(self.id)

        print("This is:" + self.getName()) # 获取线程名称

        print(‘线程结束:‘ + str(time.time()))

if __name__ == "__main__":

    t1 = MyThread(999)

    print(‘线程开始:‘+str(time.time()))

    t1.setDaemon(True)

    t1.start()

    print(‘主线程打印开始:‘+str(time.time()))

    for in range(5):

        print(i)

    time.sleep(2)

    print(‘主线程打印结束:‘ + str(time.time())) 

线程开始:1497536678.8509264
主线程打印开始:1497536678.8509264
0
1
2
3
4
主线程打印结束:1497536680.8510408

t1.setDaemon(True)的操作,将子线程设置为了守护线程。根据setDaemon()方法的含义,父线程打印内容后便结束了,不管子线程是否执行完毕了。

如果在线程启动前没有加t1.setDaemon(True),输出结果为:

线程开始:1497536865.3215919
主线程打印开始:1497536865.3215919
0
1
2
3
4
主线程打印结束:1497536867.3217063
999
This is:Thread-1
线程结束:1497536875.3221638

程序运行中,执行一个主线程,如果主线程又创建一个子线程,主线程和子线程就分兵两路,分别运行,那么当主线程完成想退出时,会检验子线程是否完成,如果子线程未完成,则主线程会等待子线程完成后再退出;

有时我们需要的是,子线程运行完,才继续运行主线程,这时就可以用join方法(在线程启动后面);

但是有时候我们需要的是,只要主线程完成了,不管子线程是否完成,都要和主线程一起退出,这时就可以用setDaemon方法(在线程启动前面)。

参考博客:http://www.cnblogs.com/UncleYong/p/6987112.html

时间: 2024-10-02 06:01:01

Python中threading的join和setDaemon的区别及用法的相关文章

python中threading模块详解(一)

python中threading模块详解(一) 来源 http://blog.chinaunix.net/uid-27571599-id-3484048.html threading提供了一个比thread模块更高层的API来提供线程的并发性.这些线程并发运行并共享内存. 下面来看threading模块的具体用法: 一.Thread的使用 目标函数可以实例化一个Thread对象,每个Thread对象代表着一个线程,可以通过start()方法,开始运行. 这里对使用多线程并发,和不适用多线程并发做

python中_、__和__xx__的区别

python中_.__和__xx__的区别 本文为译文,版权属于原作者,在此翻译为中文分享给大家. 英文原文地址:Difference between _, __ and __xx__ in Python 在学习Python时,很多人都弄不清楚各种下划线的意思,而且在这之前已经给其他人解释过很多遍了,是时候把它记录下来. "_"单下划线 Python中不存在真正的私有方法.为了实现类似于c++中私有方法,可以在类的方法或属性前加一个“_”单下划线,意味着该方法或属性不应该去调用,它并不

python中filter、map、reduce的区别

python中有一些非常有趣的函数,今天也来总结一下,不过该类的网上资料也相当多,也没多少干货,只是习惯性将一些容易遗忘的功能进行整理. lambda 为关键字.filter,map,reduce为内置函数. lambda:实现python中单行最小函数. g = lambda x: x * 2 #相当于 def g(x): return x*2 filter(function, sequence):对sequence中的item依次执行function(item),将执行结果为True的ite

python中threading的用法

摘自:http://blog.chinaunix.net/uid-27571599-id-3484048.html 以及:http://blog.chinaunix.net/uid-11131943-id-2906286.html threading提供了一个比thread模块更高层的API来提供线程的并发性.这些线程并发运行并共享内存. 下面来看threading模块的具体用法: 一.Thread的使用 目标函数可以实例化一个Thread对象,每个Thread对象代表着一个线程,可以通过sta

Python中os与sys两模块的区别

转载文章 os与sys模块的官方解释如下: os: This module provides a portable way of using operating system dependent functionality. 这个模块提供了一种方便的使用操作系统函数的方法. sys: This module provides access to some variables used or maintained by the interpreter and to functions that i

python中os模块与sys模块的区别

我们先来看下官方解释:os: This module provides a portable way of using operating system dependent functionality.这个模块提供了一种方便的使用操作系统函数的方法. sys: This module provides access to some variables used or maintained by the interpreter and to functions that interact stro

Python中元组,列表,字典的区别

Python中,有3种内建的数据结构:列表.元组和字典.1.列表     list是处理一组有序项目的数据结构,即你可以在一个列表中存储一个序列的项目.列表中的项目.列表中的项目应该包括在方括号中,这样python就知道你是在指明一个列表.一旦你创建了一个列表,你就可以添加,删除,或者是搜索列表中的项目.由于你可以增加或删除项目,我们说列表是可变的数据类型,即这种类型是可以被改变的,并且列表是可以嵌套的.实例:#coding=utf-8animalslist=['fox','tiger','ra

Python基础学习-Python中最常见括号()、[]、{}的区别

Python中最常见括号的区别: 在Python语言中最常见的括号有三种,分别是:小括号().中括号[].花括号{}:其作用也不相同,分别用来代表不同的Python基本内置数据类型. Python中的小括号(): 代表tuple元祖数据类型,元祖是一种不可变序列.创建方法很简单,大多数时候都是小括号括起来的. 1 >>> tup = (1,2,3) 2 >>> tup 3 (1, 2, 3) 4 >>> () #空元祖 5 () 6 >>&

关于python中赋值、浅拷贝、深拷贝之间区别的深入分析

大家都知道,在python中复制一个对象有多种方法,其中常用的是赋值.浅拷贝和深拷贝,这三者之间有哪些区别和哪些坑呢? 首先,定义一下: 赋值:  a =1    b =a    a赋值给了b 浅拷贝: a = []  b = a.copy() 或者import copy             b = copy.copy(a) 深拷贝:import copy  a = []   b = copy.deepcopy(a) 未完待续