iOS POSIX多线程编程

关于多线程的介绍多线程的创建使用场景Runloop可以参考《iOS多线程编程指南》。已上传到GitHub仓库。

这里主要说明线程同步技术的锁,尤其是POSIX互斥锁。已经写成了Demo,可以对照着看。GitHub地址:https://github.com/xiaoL0204/PthreadsDemo
将来也会从项目中提取更多Demo出来,对应不同的多线程知识。

这个Demo适用如下场景:同时向服务器取不同的数据,每次回调以后在子线程中处理数据(要共享数据),在主线程中显示数据。
因为数据的处理是在子线程中,在主线程UITableView reloadData显示数据;必须等到tableView刷新完成以后才能处理下一次数据回调,否则在reload data时数据源修改了会引起崩溃。

对于这样的场景,这里考虑使用条件变量进行线程间同步
原理如下:在数据处理子线程中等待条件成立,若不成立则会一直等待,直到主线程刷新完成并发出激活信号后重新激活数据处理子线程;若成立则不会等待,直接进入主线程刷新UI。

Demo 使用了如下变量和函数:

1、pthread_mutex_t
2、pthread_cond_t
3、pthread_cond_wait()
4、pthread_cond_signal()
5、pthread_join()
它们的含义和使用方法如下:

1、pthread_mutex_t 互斥锁

两种方法创建互斥锁静态方式动态方式

静态方式:

使用宏PTHREAD_MUTEX_INITIALIZER来初始化互斥锁,属性参数默认:

pthread_mutex_t mutex_t = PTHREAD_MUTEX_INITIALIZER;

动态方式:

可以指定互斥锁属性:

int pthread_mutex_init(pthread_mutex_t * __restrict,
        const pthread_mutexattr_t * _Nullable __restrict);

2、pthread_cond_t 条件变量

运用于线程间同步。一般和pthread_mutex_t一起使用。
可以使用静态方式动态方式初始化条件变量

静态方式:

用宏PTHREAD_COND_INITIALIZER来初始化静态定义的条件变量,使其具有缺省属性

动态方式:

指定条件变量的属性:

int pthread_cond_init(
        pthread_cond_t * __restrict,
        const pthread_condattr_t * _Nullable __restrict)
        __DARWIN_ALIAS(pthread_cond_init);

3、pthread_cond_wait()函数

int pthread_cond_wait(pthread_cond_t * __restrict,
        pthread_mutex_t * __restrict) __DARWIN_ALIAS_C(pthread_cond_wait);

pthread_cond_wait() 必须与pthread_mutex_t 配套使用。
用于阻塞当前线程,等待别的线程使用 pthread_cond_signal()pthread_cond_broadcast()来唤醒它。
具体来说,就是函数将解锁pthread_mutex_t指向的互斥锁,并使当前线程阻塞在pthread_mutex_t指向的条件变量上。
因此,在使用时,最好的方法是循环调用pthread_cond_wait函数,循环的终止条件为额外定义的变量。如下面核心代码中的while循环。

4、pthread_cond_signal()函数

int pthread_cond_signal(pthread_cond_t *);

作用:激活一个处于阻塞等待状态的线程,存在多个阻塞线程时按规则激活其中第一个。

pthread_cond_signal 函数会发送信号给其它阻塞在pthread_cond_t指向的条件变量的线程,阻塞在该条件变量上的线程接收信号后,脱离阻塞状态,继续执行后续代码。
使用pthread_cond_signal一般不会有“惊群现象”产生,他最多只给一个线程发信号。假如有多个线程阻塞在这个条件变量的话,会根据各阻塞线程优先级的高低确定哪个接收到信号的线程接继续执行后续代码。如果各线程优先级相同,则按入队顺序激活其中第一个pthread_cond_signal()只会激活最多一个等待该条件的线程。
pthread_cond_signal 在多处理器上可能同时唤醒多个线程,当你只能让一个线程处理某个任务时,其它被唤醒的线程就需要继续 wait。所以pthread_cond_wait()需要使用while作为外部判断。

5、pthread_join()函数

int pthread_join(pthread_t , void * _Nullable * _Nullable)
        __DARWIN_ALIAS_C(pthread_join);

使一个线程等待另一个线程结束。

其它函数:

6、pthread_cond_timedwait()函数

int pthread_cond_timedwait(
        pthread_cond_t * __restrict, pthread_mutex_t * __restrict,
        const struct timespec * _Nullable __restrict)
        __DARWIN_ALIAS_C(pthread_cond_timedwait);

函数到了一定的时间,即使条件未发生也会解除阻塞。这个时间由参数abstime指定。

7、pthread_cond_broadcast()函数

int pthread_cond_broadcast(pthread_cond_t *);

唤醒所有被pthread_cond_wait()函数阻塞在某个条件变量上的线程,pthread_cond_t指针指向这个条件变量。


主要的方法实现:

- (void)fetchHomeData{
    self.themeListArr = [NSMutableArray array];
    __weak __block typeof(self) weakSelf = self;
    
    NSMutableArray *serverThemeArr = [NSMutableArray array];
    dispatch_queue_t queue_t = dispatch_queue_create("com.dispatch.themeserial", DISPATCH_QUEUE_SERIAL);
    __block BOOL oneJobdone = YES;
    __block pthread_cond_t cond_t = PTHREAD_COND_INITIALIZER;
    //为了防止竞争,条件变量的使用总是和一个互斥锁结合在一起。
    __block pthread_mutex_t mutex_t = PTHREAD_MUTEX_INITIALIZER;
    
    [[XLDataFetchHandler sharedInstance] requestAllHomeThemeListWithCompletion:^(NSString *themeIds,NSArray *themeList, BOOL httpDone) {
        dispatch_async(queue_t, ^{
            pthread_t threadId = pthread_self();
            
            NSLog(@"requestAllHomeThemeListWithCompletion   fetch data!  themeIds:%@",themeIds);
            
            while (!oneJobdone) {  //为何使用while判断:防止可能存在的“惊群效应”。pthread_cond_wait里的线程可能会被意外唤醒,如果这个时候oneJobdone为NO,则说明UI没有刷新完成。这个时候,应该让线程继续进入pthread_cond_wait
                pthread_cond_wait(&cond_t, &mutex_t);   // pthread_cond_wait用于阻塞当前线程,等待别的线程使用 pthread_cond_signal() 或pthread_cond_broadcast来唤醒它
            }
            
            oneJobdone = NO;
            
            [serverThemeArr addObjectsFromArray:themeList];
            if (httpDone) {
                self.themeListArr = [NSMutableArray arrayWithArray:serverThemeArr];
//                [self sortThemeListArray];   //请求结束,排序
            }else{
//                [weakSelf filterOriginThemeListWithPartList:themeList];   //一次请求结束,过滤
            }
            
            
            dispatch_async(dispatch_get_main_queue(), ^{
                NSLog(@"requestAllHomeThemeListWithCompletion   reloadData before");
//table view reload,不知道什么时候结束。所以要在reload data完成后发信号
                [weakSelf.tableView reloadData];
                NSLog(@"requestAllHomeThemeListWithCompletion   reloadData after");
                oneJobdone = YES;
                //对条件变量cond_t发信号,激活一个等待该条件的线程,存在多个等待线程时按入队顺序激活其中一个
                pthread_cond_signal(&cond_t);
                NSLog(@"requestAllHomeThemeListWithCompletion   signal");
            });
            pthread_join(threadId, NULL);
        });
        
    }];
    
}

运行效果图:

效果图1、顺序执行
效果图2、阻塞执行
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 158,736评论 4 362
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,167评论 1 291
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 108,442评论 0 243
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 43,902评论 0 204
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,302评论 3 287
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,573评论 1 216
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,847评论 2 312
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,562评论 0 197
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,260评论 1 241
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,531评论 2 245
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,021评论 1 258
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,367评论 2 253
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,016评论 3 235
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,068评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,827评论 0 194
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,610评论 2 274
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,514评论 2 269

推荐阅读更多精彩内容

  • 转自:Youtherhttps://www.cnblogs.com/youtherhome/archive/201...
    njukay阅读 1,593评论 0 52
  • 简介 线程创建 线程属性设置 线程参数传递 线程优先级 线程的数据处理 线程的分离状态 互斥锁 信号量 一 线程创...
    第八区阅读 8,389评论 1 6
  • 引用自多线程编程指南应用程序里面多个线程的存在引发了多个执行线程安全访问资源的潜在问题。两个线程同时修改同一资源有...
    Mitchell阅读 1,929评论 1 7
  • 摘要 线程概念,线程与进程的区别与联系学会线程控制,线程创建,线程终止,线程等待了解线程分离与线程安全学会线程同步...
    狼之足迹阅读 443评论 2 3
  • iOS 多线程系列 -- 基础概述iOS 多线程系列 -- pthreadiOS 多线程系列 -- NSThrea...
    shannoon阅读 2,547评论 1 8