线程的生命周期_iOS线程生命周期的监控

c1e18845f47949a63fb9dd769ecc662b.png

黑客技术 点击右侧关注,了解黑客的世界! 3718f4dab93a100ad52a5e91e4163141.png

e8590231420701329920c664d6236a27.png

Linux编程 点击右侧关注,免费入门到精通! 3718f4dab93a100ad52a5e91e4163141.png

作者丨欧阳大哥2013https://www.jianshu.com/p/813ba526204b

iOS系统通过Core Services层的Foundation框架提供基于OC语言的NSThread和NSOperationQueue类来实现对线程和线程池的管理和使用。同时也提供了一套基于C语言的GCD线程池函数库来支持多线程的处理应用。这些高级的线程类或者函数的内部实现大部分最终都会调用POSIX标准中的pthread线程库中的pthread_xxx系列函数(#include )来完成线程的创建、运行、暂停、恢复、销毁、结束等操作。用户态下的线程创建通过系统调用到达内核态的BSD层并创建bsdthread对象,而BSD层则调用Mach层的ksthread对象来完成最终线程的创建和调度的。

68d5b3e3863615a09de45434fde81892.png

线程架构图

pthread库中除了提供一系列标准的线程操作API外,还提供了一个用于监控线程创建、运行、结束、销毁的内省函数(单词introspection翻译为内省,但我觉得叫拦截器可能更好一些)。这个函数定义在头文件#include 中,函数的签名为:

pthread_introspection_hook_t pthread_introspection_hook_install(pthread_introspection_hook_t hook)

函数的作用是安装一个回调函数来挂钩线程生命周期的四个过程。因此函数的入参是一个函数指针,返回的则是老的挂钩函数的指针。回调函数是一个格式为pthread_introspection_hook_t类型的函数,其格式定义如下:

typedef void (*pthread_introspection_hook_t)(unsigned int event, pthread_t thread, void *addr, size_t size);

回调函数的每个参数的意义如下:

event:指定线程所处的状态。thread: 线程的句柄,每个pthread线程都由一个pthread_t类型句柄来唯一标识。addr: 为线程分配的栈内存的基地址。size: 为线程分配的栈内存的尺寸。

上面说的每一个线程有创建、运行、终止、销毁四个状态,而event则是用来表示线程的四种状态的值,它的值是如下枚举结构的某一个值:

enum {
    PTHREAD_INTROSPECTION_THREAD_CREATE = 1,  //创建
    PTHREAD_INTROSPECTION_THREAD_START,    //运行
    PTHREAD_INTROSPECTION_THREAD_TERMINATE,  //终止
    PTHREAD_INTROSPECTION_THREAD_DESTROY,  //销毁
};

需要注意的是在内省函数中设置回调挂钩函数后只会监控设置之后的所有线程状态的变化。因此如果我们要监控整个应用生命周期的所有线程的状态时,需要尽可能早的进行回调函数的设置,比如可以在某个类的+load方法中,或者在某个全局C++对象的构造函数中设置等等。

回调挂钩函数中的第二个参数thread是一个类型为pthread_t线程句柄对象,这个对象的结构并没有对外公开。但是因为pthread库已经被苹果开源:https://opensource.apple.com/source/libpthread/  

因此我们可以通过线程句柄对象的内部定义来获取关于线程的更多信息。以方便我们能对线程的各种数据进行更加详细的记录。当然这里我们需要考虑到线程句柄的不同版本下的数据成员的问题。

最后我们实现一个简单的在main函数内实现线程监控的代码示例:

#include #include #include #include pthread_introspection_hook_t   g_oldpthread_introspection_hook = NULL;void mypthread_introspection_hook(unsigned int event, pthread_t thread, void *addr, size_t size){__uint64_t threadid;
    pthread_threadid_np(thread, &threadid);printf("thread_id = %d,  addr = %p, size = %d\n", threadid, addr, size);switch (event)
     {case PTHREAD_INTROSPECTION_THREAD_CREATE://dothing ..break;case PTHREAD_INTROSPECTION_THREAD_START://dothing ..break;case  PTHREAD_INTROSPECTION_THREAD_TERMINATE://dothing ..break;case PTHREAD_INTROSPECTION_THREAD_DESTROY://dothing ..break;
      }//记得在最后或者开头调用老的hook函数if (g_oldpthread_introspection_hook != NULL)
    g_oldpthread_introspection_hook(event, thread, addr, size);
}int main(int argc, char *argv[]){//注册线程监控的回调函数为mypthread_introspection_hook
   g_oldpthread_introspection_hook  = pthread_introspection_hook_install(mypthread_introspection_hook);
   @autoreleasepool {return UIApplicationMain(argc, argv, nil, NSStringFromClass([AppDelegate class]));
    }
}

你可以通过开源代码中对pthread_t类型结构体的定义来获取线程的更多信息,但是要注意线程库的版本信息。

线程监控回调函数中的代码应该尽可能的精简和高效,包括官方的头文件中也有一段说明(实际上是可以被appstore审核通过的):

This should only be used for introspection and debugging tools.  Do not rely

on it in shipping code.

 推荐↓↓↓ 

f9e392289d5c0cf63557b357c97ce2f6.png

?16个技术公众号】都在这里!

涵盖:程序员大咖、源码共读、程序员共读、数据结构与算法、黑客技术和网络安全、大数据科技、编程前端、Java、Python、Web编程开发、Android、iOS开发、Linux、数据库研发、幽默程序员等。

d71ed386b64791c7ae6fd57bb4c25caf.png万水千山总是情,点个 “ 在看” 行不行
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值