现在的位置: 首页 > 综合 > 正文

android bionic缺失pthread_cancel的解决方法

2018年02月15日 ⁄ 综合 ⁄ 共 1906字 ⁄ 字号 评论关闭

转自:http://blog.csdn.net/darkengine/article/details/7106207

在native code中使用多线程好处多多,但是Android的bionic并没有完全实现标准POSIX线程库的所有API,例如pthread_cancel()。但是google这样做肯定有原因,被cancel的thread不一定已经把自己拥有的资源释放掉,因此很可能带来内存泄露,锁没有释放等问题。这些问题在移动设备上更加突出。

首先介绍一个指标的方法,使用signal替代cancel调用:

当worker thread超时时,在主线程(或者是监视进程)中调用

  1. if ( (status = pthread_kill(pthread_id, SIGUSR1)) != 0)   
  2. {   
  3.     printf("Error cancelling thread %d, error = %d (%s)", pthread_id, status, strerror status));  
  4. }   

在worker thread中加入对SIGUSR1信号的处理

  1. struct sigaction actions;  
  2. memset(&actions, 0, sizeof(actions));   
  3. sigemptyset(&actions.sa_mask);  
  4. actions.sa_flags = 0;   
  5. actions.sa_handler = thread_exit_handler;  
  6. rc = sigaction(SIGUSR1,&actions,NULL);  
  7. void thread_exit_handler(int sig)  
  8. {   
  9.     printf("this signal is %d \n", sig);  
  10.     pthread_exit(0);  
  11. }  

参考自:http://stackoverflow.com/questions/4610086/pthread-cancel-alternatives-in-android-ndk

最根本的解决方法是重写worker thread,使用poll或者select等处理IO操作防止stuck的发生,下面是Android源码system/libsysutils/src/SocketListener.cpp的处理方法

1,创建worker thread前先创建通讯管道

  1. if (pipe(mCtrlPipe)) {  
  2.     SLOGE("pipe failed (%s)", strerror(errno));  
  3.     return -1;  
  4. }  
  5.   
  6. if (pthread_create(&mThread, NULL, SocketListener::threadStart, this)) {  
  7.     SLOGE("pthread_create (%s)", strerror(errno));  
  8.     return -1;  
  9. }  

2,在worker thread的大循环中使用select同时监控管道和IO fd

  1. while(1){  // 一般工作进程都带一个大循环  
  2.  FD_SET(mCtrlPipe[0], &read_fds);  
  3.         if (mCtrlPipe[0] > max)  
  4.             max = mCtrlPipe[0];  
  5.  if ((rc = select(max + 1, &read_fds, NULL, NULL, NULL)) < 0) {  
  6.             SLOGE("select failed (%s)", strerror(errno));  
  7.             sleep(1);  
  8.             continue;  
  9.         } else if (!rc)  
  10.             continue;  
  11.   
  12.  // 有人喊停了  
  13.  if (FD_ISSET(mCtrlPipe[0], &read_fds))  
  14.        break;  
  15. }  

3,需要退出时通过管道通知worker thread

  1. if (write(mCtrlPipe[1], &c, 1) != 1) {  
  2.     SLOGE("Error writing to control pipe (%s)", strerror(errno));  
  3.     return -1;  
  4. }  

抱歉!评论已关闭.