python3内threading库引入了Thread类,包含了一系列方法,其中,对于join()方法一直不太理解对于join方法,W3Cschool上的定义是:join([time]): 等待至线程中止。这阻塞调用线程直至线程的join() 方法被调用中止-正常退出或者抛出未处理的异常-或者是可选的超时发生.通过查阅一些网络上的讲解,对它大概有了一定的了解。下面通过文字加代码的形式分享一下我的观点,最后给出总结。
1.join()函数无参数时
主线程在所有引用join的线程结束后结束。
import _thread,time,threading
from threading import Thread
exitFlag = 0
class myThread (Thread):
def __init__(self, threadID, name, counter):
threading.Thread.__init__(self)
self.threadID = threadID
self.name = name
self.counter = counter
def run(self):
print ("开始线程:" + self.name)
print_time(self.name, self.counter, 5)
print ("退出线程:" + self.name)
def print_time(threadName, delay, counter):
while counter:
if exitFlag:
threadName.exit()
time.sleep(delay)
print ("%s: %s" % (threadName, time.ctime(time.time())))
print(threading.enumerate()) #展示当前进行中的线程列表,感觉显示界面太繁琐的话,此行代码可以注释掉
counter -= 1
if __name__ == '__main__':
# 创建新线程
thread1 = myThread(1, "Thread-1", 1)
thread2 = myThread(2, "Thread-2", 2)
# 开启新线程
thread1.start()
thread2.start()
thread1.join()
thread2.join() #此处,由于两个子进程都引入了join,因此主进程必须在两个子进程运行完后才能结束。
print ("退出主线程") #主线程可以理解为main的后续部分。
print("anyting you want to print")
运行结果:
在进程1,2都退出后主线程才退出.
情况二:
若在上述代码的thread2.join()前面加上#将其注释掉,则运行结果如下:
如图所示,由于没有对线程Thread2添加join函数,在线程1结束后,主线程会直接结束,线程2的剩余部分会接着继续执行完.
2.join(x)内传递了变量x,
则表示主线程在子线程执行x秒后结束,子进行接着继续执行
***为了显示对比,将最后一行的print("anyting you want to print")改为print(time.ctime(time.time()))***
一.Thread1.join(),Thread2.join(2)
对其进行分析,Thread1.join()表示主线程在线程1结束后才能结束,也就是整个程序执行5s后,Thread2.join(2)表示主线程在线程2执行2s后才能结束,也就是整个程序执行2s后结束,执行结果如下:
结果主线程既没有在线程1结束后结束,也没有在线程2执行2s后结束,而是在线程1结束后,线程2再执行2s后才结束.
因此,说明join()优先度大于join(x),即在满足join()之后再来进行join(x)的判决。
对于Thread1.join(3),Thread2.join(4),运行结果如下:
总共经历了3+4=7s后,主线程结束,也就是说在线程1执行3s,线程2再执行4s后,主线程结束.
倘若join()内变量设置的很大,比如Thread1.join(100) Thread2.join(2),那么线程1结束后线程2再执行2s主线程即可结束,并不用让线程1经历5s执行完毕后再等待95s。
总结:
1.如果所有子线程都没配置.join,则主线程开始就结束,然后逐次执行子线程
2.当join()无参数传递时,则主线程会在所有配置的join()的子线程结束后结束,随后执行未配置join的子线程
3.当部分子线程配置了无参数join(),部分配置了join(x)后,在所有配置无参数join()子线程执行完后,若还有配置join(x)的子线程未完成,则每个子线程依次执行其配置的时间后,主线程结束。(若剩余执行时间不足x,则主线程提前结束)
4.无论join()函数如何配置,设置的时间多大,一旦所有对应的子线程结束,主线程不会等大欠缺的时间,而是直接结束。