线程库实行了POSIX线程标准通常称为Pthreads。POSIX线程具有很好的可移植性,使用pthreads编写的代码可运行于Solaris、FreeBSD、Linux 等平台,Windows平台亦有pthreads-win32可供使用 。
Pthreads定义了一套C语言的类型、函数与常量,它以pthread.h头文件和一个线程库实现。
Xcode是一个包容性非常强的编辑器,不仅可以编辑本家的OC语言和swift语言,也可以编辑最底层的C语言,还有C++语言等,简直不能太强大。
使用Pthread,首先需要导入Pthread线程库。
#import <pthread.h>
点进去Pthread库,可以看到都是一些C语言的最底层的一些封装。
其实这些我们都不必太关心,如果你是大牛,也可以研习一下这些底层的东西,但对于开发来说,我们看中的还是如何使用。
创建一个子线程的函数方法是:pthread_create。
pthread_create(<#pthread_t? _Nullable *restrict _Nonnull#>, <#const pthread_attr_t *restrict _Nullable#>, <#void * _Nullable (* _Nonnull)(void * _Nullable)#>, <#void *restrict _Nullable#>)
这个函数需要四个参数,会有一个返回值。返回值是int类型的,当返回值为0时,说明成功创建了一个子线程。当返回值不为0时,说明创建子线程失败了,可以打印出来这个错误代码,上网搜错误原因。
接下来我们最关心的还是需要的四个参数,现在具体说下这四个参数:
参数1.指向线程代号的指针
参数2.线程的属性
参数3.指向函数的指针
参数4.传递给第三个参数中的函数的参数
参数1是指向线程代号的指针,是pthread_t类型的id的指针。创建一个pthread_t类型的id如下:
pthread_t pthreadId;//是一个结构体structure
那么,参数1需要传入的值就是:&pthreadId
参数2是线程的属性,可以不用管,直接传入一个:NULL
参数3是指向函数的指针,这个参数的类型是:void * (*)(void *),“void *”表示的是函数的返回值,“(*)”表示的是函数指针,“(void *)”表示的是函数的参数。在C语言中,void *类型,就相当于OC语言中的id类型,即任意类型。
那么参数3需要传入的就是:参数为任意类型,返回值也是任意类型的一个函数的指针。好吧,我们就定义一个这样的函数。
//根据参数三的类型定义一个函数
void * demo(void *params) {
return NULL;
}
这个函数是符合要求的,参数是任意类型的params,返回值是NULL,参数3传入:&demo即可。
参数4是上面定义的demo函数的参数,也就是传入的params,是一个任意类型的,我们可以随便传入一个字符串,譬如:NSString *str = @"Hello World";
另外,请注意,如果你用的MRC机制,参数4直接传入的就是:str,但是如果你用的是ARC机制,参数4传入“str”之后,会报错,并提示你正确的写法:(__bridge void *)(str)。
这是什么原因呢?原因很简单,在 ARC 开发中,如果涉及到和C语言中的相同的数据类型进行转换,需要使用 __bridge “桥接”,因为ARC是自动处理机制,但是只负责OC的处理,不引入桥接,ARC不处理C语言的内容。不过这个不需要记,遇到这种情况,会自动提示正确写法的,直接按照正确写法,就OK了。
我们的子线程已经创建好了,没有开玩笑,真的已经创建好了哟。
如果你不信,那就在demo函数中打印当前的线程,同时把params参数的值也给打印一下。
NSLog(@"线程:%@, 参数:%@", [NSThread currentThread],params);
打印出来的结果是这样的:
2017-04-01 10:20:48.225 pThread[1970:66224] 线程:{number = 3, name = (null)}, 参数:Hello World
看到了吧,线程number不是1,说明不是主线程,参数就是刚才我们传入的str字符串。
除了在demo函数方法里是子线程的处理,其它地方写的代码,还是由主线程处理。
在此特别提醒,多线程开发中,千万不要相信一次的执行结果?。?!这个可以自己去试,譬如:在主线程中打印创建线程函数的返回值(就是pthread_create的返回值,数据类型为int类型),在子线程中打印当前线程和demo函数的参数params,你永远不知道是先打印主线程中的返回值,还是先打印子线程中的当前线程和参数params,这个需要多次试验,才能看出来效果。所以再次提醒:多线程开发中,千万不要相信一次的执行结果!??!