python-多线程

示例

def music(name):
    print '{} start listening to music'.format(name)
    time.sleep(2)  # 模拟遇到io
    print "{} end".format(name)


def movie(name):
    print '{} start watching movie'.format(name)
    time.sleep(4)  # 模拟遇到io
    print "{} end".format(name)


if __name__ == '__main__':
    t1 = threading.Thread(target=music, args=('tom',))  # Thread是一个类,实例化产生t1对象,这里就是创建了一个线程对象t1
    t1.start()  # 线程执行
    t2 = threading.Thread(target=movie, args=('jack',))  # 这里就是创建了一个线程对象t2
    t2.start()
    # 如果子线程不加join,则主线程会直接执行print,但主线程不会结束
    # t1.join()  # 执行join,则会等t1子线程结束,主线程才执行后面的print操作
    # t2.join()  # 同理
    print "...to the end..."
    # 不论有无join, 都需要等待非守护子线程结束之后,主线程才结束。

结果:

tom start listening to music
jack start watching movie
…to the end…
tom end
jack end

说明:
1.使用threading.Thread创建一个线程对象,相当于在主线程之外开了一个分支,同时进行。
2.主线程和music, movie两个子线程同时start,主线程运行到print,但并未结果,需要等待所有非守护子线程结束,主线程才结束
3.sleep()模拟io,经过2s后music结束,打印。同理,经过4s后movie结束,打印。

那么,当我们希望把主线程的print '…to the end…'放到最后执行,即等到所有子线程完成之后,再执行print,那么就需要使用join()

join

在子线程完成运行之前,这个子线程的父线程将一直被阻塞。

if __name__ == '__main__':
    t1 = threading.Thread(target=music, args=('tom',))  # Thread是一个类,实例化产生t1对象,这里就是创建了一个线程对象t1
    t1.start()  # 线程执行
    t2 = threading.Thread(target=movie, args=('jack',))  # 这里就是创建了一个线程对象t2
    t2.start()
    # 如果子线程不加join,则主线程会直接执行print,但主线程不会结束
    t1.join()  # 执行join,则会等t1子线程结束,主线程才执行后面的print操作
    # t2.join()  # 同理
    print "...to the end..."
    # 不论有无join, 都需要等待非守护子线程结束之后,主线程才结束。

说明:
1.t1.join(): 在t1即music子线程完成之前,阻塞,直到t1子线程完成,才执行后面的print语句。

守护进程setDaemon

  • 如果某个子线程设置为守护线程,主线程其实就不用管这个子线程了,当所有其他非守护线程结束,主线程就会退出。
  • setDaemon必须在start()之前
  • 使用场景:只要主线程完成了,不管子线程是否完成,都要和主线程一起退出。

if __name__ == '__main__':
    t1 = threading.Thread(target=music, args=('tom',))  # Thread是一个类,实例化产生t1对象,这里就是创建了一个线程对象t1
    t1.start()  # 线程执行
    t2 = threading.Thread(target=movie, args=('jack',))  # 这里就是创建了一个线程对象t2
    t2.setDaemon(True)  # 设置t2为守护进程的话,主线程就不管t2的运行状态,只等待t1结束,主线程就结束。 因为t1为2s,t2为4s,故t2不会结束,不打印。
    t2.start()
    # 如果子线程不加join,则主线程会直接执行打印,但主线程不会结束
    print "...to the end..."

其他方法

run(): 线程被cpu调度后自动执行线程对象的run方法
start():启动线程活动。
isAlive(): 返回线程是否活动的。
getName(): 返回线程名。
setName(): 设置线程名。
threading模块提供的一些方法:
threading.currentThread(): 返回当前的线程变量。
threading.enumerate(): 返回一个包含正在运行的线程的list。正在运行指线程启动后、结束前,不包括启动前和终止后的线程。
threading.activeCount():返回正在运行的线程数量,与len(threading.enumerate())有相同的结果。

参考文章

1、https://www.jianshu.com/p/59c236746b5c
2、多进程:
https://www.liaoxuefeng.com/wiki/897692888725344/923056295693632

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值