在最近开发的多线程程序中,观察到一种现象,线程调用pthread_exit()退出后,进程的VSZ没有减少,随着这样的线程增多,可以看到VSZ的值变得越来越大。
一开始以为是程序那里漏内存,查看了所有new的地方,没有发现有漏内存的情况。
通过pmap分析,发现跟没有线程退出情况的进程相比,会多出下面几个内存块,其他部分都没有不同的地方。
pmap 19661
...................
00007f80eeb5c000 4K ----- [ anon ]
00007f80eeb5d000 8192K rwx-- [ anon ]
................................
gdb里头从这些地址里头看不到任何有意义的内容
通过valgrind也没发现问题
valgrind --tool=memcheck --leak-check=full -v --track-origins=yes --log-file=val.log --track-fds=yes --time-stamp=yes --show-reachable=yes my_app
于是就怀疑是线程退出的时候没有释放资源
从网上查找资源看到chinaunix上面有些文章,关于thread的资源安全释放的问题告诫如下:
如果线程是joinable的,主线程(或某个负责回收线程的线程)需要调用pthread_join()来回收线程
如果不想把回收线程阻塞住,而让系统自动回收线程资源,即不调用pthread_join(),则线程必须是detached。
joinable和detached是通过pthread_attr_setdetachstate()来设置的。
由于我的回收线程还需要处理别的事务不能长时间阻塞住,并且通过打印pthread_join()前后的时间差发现即使线程已经退出,pthread_join()仍然可能会等上5秒钟,
所以最后采用的是pthread_exit() + detached的方法,而不是pthread_exit() + pthread_join().
回头再来看看为什么是4k和8M
首先下载glibc,在nptl目录下面能找到pthread_create.c