在涉及python多线程编程的时候,有四个特别重要的概念需要大家深入理解:主线程、子线程、守护线程以及线程的join方法。文章默认大家都知道python 多线程的常规用法,这里就不再详细讨论了,仅对这四个概念进行详细描述讲解。本文可能是全网讲解相关概念最通俗易懂的文章之一了,为了加深理解大家可以拷贝代码进行实际操作,相信效果会更好!
主线程和子线程主线程:应用程序运行即为主线程(从程序第一行到最后一行执行完毕,中间遇到子线程的start,子线程去执行它的函数,主线程继续往下执行其他语句)
用户线程(子线程):在主线程中可以创建和启动新线程,默认为用户线程(子线程)
示例代码(后面关于守护线程和join方法的讲解也是基于该实例)
import threadingimport timedef test_thread(): while True: print("子线程运行!n") time.sleep(1)if __name__ == '__main__': t = threading.Thread(target=test_thread) t.start() print('主线程运行..') time.sleep(0.6) print('主线程运行...') time.sleep(0.6) print('主线程运行...') time.sleep(0.6) print('主线程运行...') time.sleep(0.6) print('主线程运行完毕')
运行结果如下:虽然主线程运行完毕,但是子线程会一直运行下去(这是个死循环!)
子线程运行!主线程运行..主线程运行...子线程运行!主线程运行...主线程运行...子线程运行!主线程运行完毕子线程运行!子线程运行!子线程运行!子线程运行!子线程运行!子线程运行!
守护线程daemon线程:守护线程,优先级别最低,一般为其它线程提供服务。通常,daemon线程体是一个无限循环。如果所有的非daemon线程(主线程以及子线程)都结束了,daemon线程自动就会终止。t.daemon 属性,设置为True,则为daemon线程,必须在start()之前调用
在上面的代码中t.start()前添加代码 t.daemon = True
运行结果如下,发现当主线程结束后,并且没有任何非守护线程继续运行时,主线程可以正常终止退出了。
子线程运行!主线程运行..主线程运行...子线程运行!主线程运行...主线程运行...子线程运行!主线程运行完毕
这里需要注意几点:
1、必须在启动之前将线程配置为守护程序或非守护程序,否则Python将引发运行时错误;
2、最后守护程序线程不会像普通线程一样正常退出,当程序中的所有非守护程序线程都完成执行时,任何剩余的守护程序线程将在Python退出时被放弃,在设计守护线程时,需要确保在主线程退出时不会产生任何负面影响。
Join方法join([time])方法的作用是等待至线程终止。它会阻塞调用线程(主线程)执行直至子线程的join() 方法被调用中止-正常退出或者抛出未处理的异常-或者是可选的超时发生。
总结:
1.join的作用是阻塞主线程,使其无法执行join以后的语句
2.多线程join的情况下,依次执行各线程的join方法,前一个结束了才能执行后一个。
3、join无参数,等待到该线程结束,才开始执行下一个线程的join。
4、join设置参数后,等待该线程指定长时间(而该线程并没有结束)。超过这个时间后就可以执行后面的语句了。
在示例代码中 t.start() 后面添加 t.join(),然后执行代码 ,运行结果如下:
子线程运行!子线程运行!子线程运行!子线程运行!子线程运行!子线程运行!子线程运行!子线程运行!子线程运行!子线程运行!子线程运行!子线程运行!子线程运行!
我们发现以下代码不会被运行,主线程一直会等待子线程的运行,这同样也是一个死循环!
print('主线程运行..') time.sleep(0.6) print('主线程运行...') time.sleep(0.6) print('主线程运行...') time.sleep(0.6) print('主线程运行...') time.sleep(0.6) print('主线程运行完毕')
我们设置t.join(2)
运行结果如下所示,我们发现主线程会等子线程2秒,然后主线程运行完毕,接下来再次进入到子线程的无限循环!
子线程运行!子线程运行!主线程运行..子线程运行!主线程运行...子线程运行!主线程运行...主线程运行...子线程运行!主线程运行完毕子线程运行!子线程运行!子线程运行!子线程运行!子线程运行!
当我们执行以下代码
t.daemon = Truet.start()t.join(2)
运行结果是:
子线程运行!子线程运行!主线程运行..子线程运行!主线程运行...子线程运行!主线程运行...主线程运行...子线程运行!主线程运行完毕
这个运行结果是否符合大家的预期呢?如果通过本文大家还没有理解python中,主线程,子线程,守护线程以及jion方法的概念,欢迎留言讨论!