ST源码分析-st_thread_join - 弦外之音

/ 0评 / 0

本文主要讲解, st_thread_join() 函数的使用 ,参考文档《st_thread_join》。下面用一个小的示例代码,演示一下 st_thread_join() 函数 的使用。

#include <stdio.h>
#include <memory.h>
#include "st.h"
#include "../common.h"
#include <stdlib.h>
​
int do_something(void *arg) {
    st_utime_t time_now = st_utime();
    printf("do_something %lld \r\n", time_now);
    return 8;
}
​
int main(int argc, char *argv[]) {
    if (st_init() < 0) {
        perror("st_init");
        exit(1);
    }
​
    st_utime_t time_now = st_utime();
    printf("start %lld\r\n", time_now);
​
    _st_thread_t *pid = st_thread_create((void *) do_something, NULL, 1, 0);
    if (pid == NULL) {
        perror("st_thread_create");
        exit(1);
    }
​
    void *return_num;
    st_thread_join(pid, &return_num);
    printf("get thread num %d\r\n", return_num);
​
    st_thread_exit(NULL);
​
    /* NOTREACHED */
    return 1;
}

运行结果如下图:

从上图可以看到,始祖协程 等待子协程运行结束,然后获取到子协程的退出码。



下面就来讲讲,这里面的原理。如下,st_thread_create() 的时候,要指定这个协程是否 可以 被别的协程 join

从上图可以看到,创建协程的时候,同时创建了一个条件变量,不熟悉条件变量的请看《ST源码分析-协程通信》。

接下来看下 st_thread_join() 函数的实现。实际上到这里,结果已经很明显了,肯定是 st_thread_join()会等待那个条件变量,然后子协程退出就会调 st_cond_signal() 通知 始祖协程。

下面就看看看是否是上面猜测的那样。

从上图看来,果然如此, st_thread_join() 阻塞在 st_cond_timedwait() 里面了,下面就来看看什么时候会被激活。如下图,子协程执行到 st_thread_exit() 的时候就会激活 始祖协程的 st_thread_join()

上图有几个重点:

1,st_thread_exit() 会把子协程变成 僵尸协程,然后 st_cond_signal() 唤醒 始祖协程。

2,始祖协程 醒来之后,就会把 子协程从 僵尸队列 剔除。

这里有一个情况要注意,如果 始祖协程,不执行 st_thread_join(),那子协程就会一直在僵尸协程队列里面,占用着内存。打印 ZOMBIEQ 就能查看当前系统有多少个僵尸协程。通常如果代码没写错,没有忘记写 st_thread_join(),是不会出现僵尸协程一直在队列的。


相关阅读:

  1. 《st_thread_join》

由于笔者的水平有限, 加之编写的同时还要参与开发工作,文中难免会出现一些错误或者不准确的地方,恳请读者批评指正。如果读者有任何宝贵意见,可以加我微信 Loken1。QQ:2338195090。

发表回复

您的电子邮箱地址不会被公开。 必填项已用 * 标注