1.背景
代码中线程池使用的std::list
管理,有并发的pop_front
和push_back
操作,都有加锁;
另外还有个服务指标监控线程,会定时调用size()
判断线程池大小,未加锁。
2.原因
在c++11之前,std::list::size
使用的是遍历方式计算大小,即从begin()
开始,迭代到end()
,得出list
大小;在c++11之后使用单独变量计数,但是需要给编译器传参 -D_GLIBCXX_USE_CXX11_ABI=1
才能开启特性(看编译器版本,有的是默认开启)。我的程序中虽然是c++11标准,但是未开启特性,导致在size()
遍历过程中,和pop_front()
push_back
出现指针异常,从而出现段错误,致使程序崩溃。
3.解决
- 在使用
std::list::size()
时也要加锁处理。 - 给编译器传参
-D_GLIBCXX_USE_CXX11_ABI=1
,但是需要注意其他依赖动态库是否异常,如果有异常需要加上此参数重编异常的动态库。