V2EX = way to explore
V2EX 是一个关于分享和探索的地方
Sign Up Now
For Existing Member  Sign In
推荐学习书目
Learn Python the Hard Way
Python Sites
PyPI - Python Package Index
http://diveintopython.org/toc/index.html
Pocoo
值得关注的项目
PyPy
Celery
Jinja2
Read the Docs
gevent
pyenv
virtualenv
Stackless Python
Beautiful Soup
结巴中文分词
Green Unicorn
Sentry
Shovel
Pyflakes
pytest
Python 编程
pep8 Checker
Styles
PEP 8
Google Python Style Guide
Code Style from The Hitchhiker's Guide
chaleaoch
V2EX  ›  Python

Python 中线程的 is_alive() 返回 Flase 意味着 线程资源已经被回收了吗?

  •  
  •   chaleaoch · Dec 16, 2020 · 3374 views
    This topic created in 1958 days ago, the information mentioned may be changed or developed.

    线程资源已经被销毁?

    它的子线程会被自动回收吗?

    以下代码输出

    test1
    aps
    irr
    irr
    irr
    

    没想明白为什么.

    代码如下

    import threading
    import time
    
    
    def irr():
        print('irr')
    
    
    def aps():
        print('aps')
        while True:
            time.sleep(6)
            irr()
    
    
    def view():
        t2 = threading.Thread(target=aps)
        t2.daemon = True
        t2.start()
        print('test1')
    
    
    t = threading.Thread(target=view)
    t.daemon = True
    t.start()
    while True:
        time.sleep(1)
    
    
    
    10 replies    2020-12-17 18:06:01 +08:00
    weyou
        1
    weyou  
       Dec 17, 2020 via Android
    输出没问题啊,后面是一直输出 irr 。
    chaleaoch
        2
    chaleaoch  
    OP
       Dec 17, 2020
    @weyou 给我感觉好像 python 没有 父子线程的概念. 父线程退出 子线程还活着.
    lanshee
        3
    lanshee  
       Dec 17, 2020
    小白问下,这个父线程退出了嘛?
    cbiqih
        4
    cbiqih  
       Dec 17, 2020
    设置了 daemon = True, 则为守护进程, 要等所有子进程退出了, 父进程才会结束的~
    4everLoveU
        5
    4everLoveU  
       Dec 17, 2020
    主线程显然不会退出啊,设置守护线程,必须等待所有守护线程结束之后才会退出
    chaleaoch
        6
    chaleaoch  
    OP
       Dec 17, 2020
    @4everLoveU
    @cbiqih

    这个代码是 主线程 起了一个 子线程 子线程起了一个孙线程.

    现在 子线程结束释放了 但是孙线程还在.

    我的问题是, 为什么孙线程没有被强制释放.


    后来调查了一下.好像 python 里面没有父子概念, 只有主线程和非主线程.

    上面代码里面的例子, 那两个线程实际上是兄弟关系. 所以 view 线程跑完了 aps 还活着.
    xx6412223
        7
    xx6412223  
       Dec 17, 2020   ❤️ 1
    楼上解释的都是小问题:
    非守护线程一般作为 main thread, 在非守护线程挂掉之前,程序不会退出(排除异常场景)
    daemon =true 是守护线程,程序退出后他们也随着挂了,所以一般用作垃圾回收,心跳等工作。
    xx6412223
        8
    xx6412223  
       Dec 17, 2020   ❤️ 1
    补充一下,除了语言层有特殊设定,线程的生命周期没有父子关系
    4everLoveU
        9
    4everLoveU  
       Dec 17, 2020   ❤️ 1
    线程的父对象都是 Thread,所有基于 Thread 创建的线程都是平等的,逻辑上看着是多重派生,实际上不存在什么父子关系,每个线程都是在自己的隔离环境中运行的

    主线程就是当前脚本的运行线程及其环境,t/t2 都是主线程中的守护线程,不存在 t2 是 t 线程中的一个隔离线程,所以你说的兄弟关系比较恰当。
    weyou
        10
    weyou  
       Dec 17, 2020   ❤️ 1
    @chaleaoch 线程的行为是操作系统决定的, 而不是语言. 各种语言的线程库只是对操作系统线程的封装.

    线程只有主线程和子线程的概念, 没听说过"父线程"的说法. 通过子线程创建的子线程和子线程本身并没有本质区别. 所以你这里的父线程(非主线程)退出子线程还活着. 而主线程退出的话, 分两种情况:
    a.) 如果此时还有非 daemon 的子线程存在, 主线程就会继续等待, 直到所有非 daemon 子线程退出才会退出.
    b.) 如果此时只剩下 daeomon 子线程, 所有的 daemon 子线程都被强制结束掉, 然后主线程退出.
    About   ·   Help   ·   Advertise   ·   Blog   ·   API   ·   FAQ   ·   Solana   ·   5315 Online   Highest 6679   ·     Select Language
    创意工作者们的社区
    World is powered by solitude
    VERSION: 3.9.8.5 · 37ms · UTC 09:41 · PVG 17:41 · LAX 02:41 · JFK 05:41
    ♥ Do have faith in what you're doing.