iOS多线程

并发编程推荐文章objc中国的并发编程API挑战

pthread

POSIX线程(POSIX threads),简称Pthreads,是线程的POSIX标准。该标准定义了创建和操纵线程的一整套API。在类Unix操作系统(Unix、Linux、Mac OS X等)中,都使用Pthreads作为操作系统的线程。是一套C语言写的线程库.

先要包含其头文件

#import <pthread.h>
- (void)viewDidLoad {
    [super viewDidLoad];
    [self createPThread];
}

- (void)createPThread{
    pthread_t pthreadID;
    NSString *param = @"piaojin";
    //pthreadID线程的标识符
    //cFunc要调用的方法(C语言的方法)
    //param传递的参数
    int result = pthread_create(&pthreadID, NULL, &cFunc, (__bridge void *)(param));
    printf("%d",result);
}

void * cFunc(void *param){
    NSLog(@"%@,param:%@",[NSThread currentThread],param);
    return NULL;
}

看代码就会发现他需要 c语言函数,这是比较蛋疼的,更蛋疼的是你需要手动处理线程的各个状态的转换即管理生命周期,比如,这段代码虽然创建了一个线程,但并没有销毁。

NSThread

这套方案是经过苹果封装后的,并且完全面向对象的。所以你可以直接操控线程对象,非常直观和方便。但是,它的生命周期还是需要我们手动管理,所以这套方案也是偶尔用用,比如 [NSThread currentThread],它可以获取当前线程类,你就可以知道当前线程的各种属性,用于调试十分方便。下面来看看它的一些用法。

创建并启动

先创建线程类,再启动

  // 创建
  NSThread *thread = [[NSThread alloc] initWithTarget:self selector:@selector(run:) object:nil];

  // 启动
  [thread start];

  [NSThread detachNewThreadSelector:@selector(run:) toTarget:self withObject:nil];

//使用 NSObject 的方法创建并自动启动

  [self performSelectorInBackground:@selector(run:) withObject:nil];

//除了创建启动外,NSThread 还以很多方法,下面我列举一些常见的方法,当然我列举的并不完整,更多方法大家可以去类的定义里去看。

//取消线程
- (void)cancel;

//启动线程
- (void)start;

//判断某个线程的状态的属性
@property (readonly, getter=isExecuting) BOOL executing;
@property (readonly, getter=isFinished) BOOL finished;
@property (readonly, getter=isCancelled) BOOL cancelled;

//设置和获取线程名字
-(void)setName:(NSString *)n;
-(NSString *)name;

//获取当前线程信息
+ (NSThread *)currentThread;

//获取主线程信息
+ (NSThread *)mainThread;

//使当前线程暂停一段时间,或者暂停到某个时刻
+ (void)sleepForTimeInterval:(NSTimeInterval)time;
+ (void)sleepUntilDate:(NSDate *)date;

GCD

Grand Central Dispatch,听名字就霸气。它是苹果为多核的并行运算提出的解决方案,所以会自动合理地利用更多的CPU内核(比如双核、四核),最重要的是它会自动管理线程的生命周期(创建线程、调度任务、销毁线程),完全不需要我们管理,我们只需要告诉干什么就行。同时它使用的也是 c语言,不过由于使用了 Block(Swift里叫做闭包),使得使用起来更加方便,而且灵活。所以基本上大家都使用 GCD 这套方案,老少咸宜,实在是居家旅行、杀人灭口,必备良药。不好意思,有点中二,咱们继续。

任务和队列

在 GCD 中,加入了两个非常重要的概念: 任务 和 队列。

任务:即操作,你想要干什么,说白了就是一段代码,在 GCD 中就是一个 Block,所以添加任务十分方便。任务有两种执行方式: 同步执行 和 异步执行,他们之间的区别是 是否会创建新的线程。

同步(sync)和异步(async)的主要区别在于会不会阻塞当线程,直到Block中的任务执行完毕!如果是 同步(sync)操作,它会塞当前线程并等待Block中的任务执行完毕,然后当前线程才会继续往运行。如果是异步(async)操作,当前线程会直接往下执行,它不会阻塞当前线程。

队列:用于存放任务。一共有两种队列, 串行队列 和 并行队列。

放到串行队列的任务,GCD 会 FIFO(先进先出) 地取出来一个,执行一个,然后取下一个,这样一个一个的执行。

放到并行队列的任务,GCD 也会 FIFO的取出来,但不同的是,它取出来一个就会放到别的线程,然后再取出来一个又放到另一个的线程。这样由于取的动作很快,忽略不计,看起来,所有的任务都是一起执行的。不过需要注意,GCD 会根据系统资源控制并行的数量,所以如果任务很多,它并不会让所有任务同时执行。

//同步任务 + 并发队列(任务还是以同步的形式,一个一个完成),没有开启新线程
- (void)sync{
    //创建并发队列DISPATCH_QUEUE_CONCURRENT,默认是同步队列DISPATCH_QUEUE_SERIAL,DISPATCH_QUEUE_PRIORITY_DEFAULT全局队列
    dispatch_queue_t queue = dispatch_queue_create("同步任务 + 并发队列", DISPATCH_QUEUE_CONCURRENT);
    dispatch_sync(queue, ^{
        for(int i = 0;i < 5;i++){
            NSLog(@"currentThread:%@,->1<-%d",[NSThread currentThread],i);
        }
    });
    
    dispatch_sync(queue, ^{
        for(int i = 0;i < 5;i++){
            NSLog(@"currentThread:%@,->2<-%d",[NSThread currentThread],i);
        }
    });
    
    dispatch_sync(queue, ^{
        for(int i = 0;i < 5;i++){
            NSLog(@"currentThread:%@,->3<-%d",[NSThread currentThread],i);
        }
    });
}

//异步任务 + 串行队列(任务还是以同步的形式,一个一个完成),开启了新线程
- (void)async{
    //创建并发队列DISPATCH_QUEUE_CONCURRENT,默认是同步队列DISPATCH_QUEUE_SERIAL,DISPATCH_QUEUE_PRIORITY_DEFAULT全局队列
    dispatch_queue_t queue = dispatch_queue_create("异步任务 + 串行队列", DISPATCH_QUEUE_SERIAL);
    dispatch_async(queue, ^{
        for(int i = 0;i < 5;i++){
            NSLog(@"currentThread:%@,->1<-%d",[NSThread currentThread],i);
        }
    });
    
    dispatch_async(queue, ^{
        for(int i = 0;i < 5;i++){
            NSLog(@"currentThread:%@,->2<-%d",[NSThread currentThread],i);
        }
    });
    
    dispatch_async(queue, ^{
        for(int i = 0;i < 5;i++){
            NSLog(@"currentThread:%@,->3<-%d",[NSThread currentThread],i);
        }
    });
}

#define ImageUrl @"http://upload-images.jianshu.io/upload_images/3362328-bdcbda41eac5e8a2.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240"
//线程间通讯,NSThread利用PerformSelector系列方法
- (void)communication{
    __weak typeof(self) weakSelf = self;
    dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
        //耗时操作,此处为从网络获取一直图片
        //耗时的操作
        NSData *imageData = [NSData dataWithContentsOfURL:[NSURL URLWithString:ImageUrl]];
        UIImage *image = [UIImage imageWithData:imageData];
        //把image传递给主线程(线程间的通讯)
        dispatch_async(dispatch_get_main_queue(), ^{
            //到主线程更新UI
            weakSelf.imageView.image = image;
        });
    });
}

//延时
- (void)delay{
    //NSObject的延时方法
    [self performSelector:@selector(run) withObject:nil afterDelay:2.0];
    
    //延时两秒
    dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(2.0 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
        [self run];
    });
    
    //利用定时器也可以达到延时执行的效果,这时候repeats要设置为NO,只执行一次,用scheduledTimerWithTimeInterval方法创建的定时器会默认添加到当前的RunLoop中去,并且启动
    [NSTimer scheduledTimerWithTimeInterval:2.0 target:self selector:@selector(run) userInfo:nil repeats:NO];
}

- (void)run{
    NSLog(@"延时两秒");
}

//一次性操作
- (void)once{
    static dispatch_once_t onceToken;
    dispatch_once(&onceToken, ^{
        NSLog(@"一次性操作");
    });
}

//多次操作
- (void)apply{
    /**
     *size_t iterations执行的次数
     **/
    dispatch_apply(10, dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^(size_t index) {
        //index每次遍历的下标,注意并不是按顺序的
        NSLog(@"%zu",index);
    });
}

//队列组操作
- (void)group{
    dispatch_group_t group = dispatch_group_create();
    //异步的组操作
    dispatch_group_async(group, dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
        for (int i = 0; i < 100; i++) {
            NSLog(@"queue1:%d",i);
        }
    });
    
    dispatch_group_async(group, dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
        for (int i = 0; i < 100; i++) {
            NSLog(@"queue2:%d",i);
        }
    });
    
    dispatch_group_async(group, dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
        NSLog(@"queue3");
    });
    
    //前面创建了三个任务,这边三个任务完成时回执行dispatch_group_notify中的block代码
    dispatch_group_notify(group, dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
        NSLog(@"三个任务都完成了!");
    });
}

当dispatch_group_async的block里面执行的是异步任务,如果还是使用上面的方法你会发现异步任务还没跑完就已经进入到了dispatch_group_notify方法里面了,这时用到dispatch_group_enter和dispatch_group_leave就可以解决这个问题.进入前dispatch_group_enter(group),离开执行完毕时dispatch_group_leave(group).

NSOperation和NSOperationQueue

NSOperation 是苹果公司对 GCD 的封装,完全面向对象,所以使用起来更好理解。 大家可以看到 NSOperation 和 NSOperationQueue 分别对应 GCD 的 任务 和 队列

NSOperation是一个抽象的类,并不具备封装操作的能力,必须使用它的子类:

  • NSInvocationOperation

  • NSBlockOperation

  • 自定义子类继承NSOperation是一个抽象的类,重写内部相应的方法(main方法)

操作步骤也很好理解:

  • 先将需要执行的操作封装到一个NSOperation对象中

  • 将NSOperation对象添加到NSOperationQueue中

  • 系统会自动将NSOperationQueue中的NSOperation取出来

  • 将取出来的NSOperation封装的操作放到一条线程中去执行

NSOperationQueue的作用

如果将NSOperation添加到NSOperationQueue中,系统会默认异步执行NSOperation的操作

添加NSOperation到NSOperationQueue中

  • 调用addOperation:的方法

  • 调用addOperationWithBlock:方法

//自定义继承NSOperation
@implementation PJCustomOperation

//需要重写main方法,任务是在main方法中执行的
- (void)main{
    NSLog(@"PJCustomOperation:%@",[NSThread currentThread]);
}

@end

没有添加到NSOperationQueue的情况

//NSInvocationOperation
- (void)invocationOperation{
    //默认在主线程中执行,并且不会创建新的线程number = 1(主线程),相单与[self performSelector:@selector(run) withObject:nil];,只有放到NSOperationQueue才会开启新线程,异步执行
    NSInvocationOperation *invocationOperation = [[NSInvocationOperation alloc] initWithTarget:self selector:@selector(run) object:nil];
    //开启
    [invocationOperation start];
}

//NSBlockOperation
- (void)blockOperation{
    //默认在主线程中执行,并且不会创建新的线程number = 1(主线程),相单与[self performSelector:@selector(run) withObject:nil];
    NSBlockOperation *blockOperation = [NSBlockOperation blockOperationWithBlock:^{
        NSLog(@"blockOperation1%@",[NSThread currentThread]);
    }];
    
    //添加额外任务会开启新线程,number != 1,在新线程中完成
    [blockOperation addExecutionBlock:^{
        NSLog(@"blockOperation2%@",[NSThread currentThread]);
    }];
    
    [blockOperation start];
}

//自定义NSOperation,需要重写main方法,任务是在main方法中执行的
- (void)customOperation{
    //此时是在主线程中执行,并没有开启新线程
    PJCustomOperation *customOperation = [[PJCustomOperation alloc] init];
    [customOperation start];
}

- (void)run{
    NSLog(@"执行任务%@",[NSThread currentThread]);
}

添加到NSOperationQueue的情况

- (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event{
    //创建队列,默认创建的是并行队列
    NSOperationQueue *queue = [[NSOperationQueue alloc] init];
    //GCD的队列形式
    //全局 主队列 串行队列 并行队列
    
    //NSOperationQueue队列形式
    //主队列 自己创建的队列
    
    //创建NSOperation
    NSInvocationOperation *invocationOperation1 = [[NSInvocationOperation alloc] initWithTarget:self selector:@selector(run:) object:@"invocationOperation1"];
    //放入队列中,就会执行不用显示的去[invocationOperation start],会创建一个新线程,异步执行
    [queue addOperation:invocationOperation1];
    
    NSInvocationOperation *invocationOperation2 = [[NSInvocationOperation alloc] initWithTarget:self selector:@selector(run:) object:@"invocationOperation2"];
    //放入队列中,就会执行不用显示的去[invocationOperation start],会创建一个新线程,异步执行
    [queue addOperation:invocationOperation2];
    
    //会创建一个新线程,异步执行
    [queue addOperationWithBlock:^{
        NSLog(@"addOperationWithBlock的形式:thread:%@",[NSThread currentThread]);
    }];

}

- (void)run:(NSString *)name{
    NSLog(@"name:%@,thread:%@",name,[NSThread currentThread]);
}

NSOperation通信与依赖关系

//利用NSOperationQueue进行现场间通讯
- (void)queueMessage{
    //利用NSOperation和NSOperationQueue来执行
    __weak typeof(self) weakSelf = self;
    NSOperationQueue *queue = [[NSOperationQueue alloc] init];
    [queue addOperationWithBlock:^{
        //耗时的操作
        NSData *imageData = [NSData dataWithContentsOfURL:[NSURL URLWithString:ImageUrl]];
        //耗时操作执行完后会到主队列执行UI更新操作
        [[NSOperationQueue mainQueue] addOperationWithBlock:^{
            weakSelf.imageView.image = [UIImage imageWithData:imageData];
        }];
    }];
}

//依赖 如果A依赖与B,则A等到B完成了才执行
- (void)dependency{
    NSOperationQueue *queue = [[NSOperationQueue alloc] init];
    
    NSBlockOperation *operation1 = [NSBlockOperation blockOperationWithBlock:^{
        NSLog(@"operation1:%@",[NSThread currentThread]);
    }];
    
    NSBlockOperation *operation2 = [NSBlockOperation blockOperationWithBlock:^{
        NSLog(@"operation2:%@",[NSThread currentThread]);
    }];
    
    NSBlockOperation *operation3 = [NSBlockOperation blockOperationWithBlock:^{
        NSLog(@"operation3:%@",[NSThread currentThread]);
    }];
    
    //添加依赖 前者(operation1)依赖后者(operation2),后者(operation2)先执行,在执行前者(operation1)
    //任务之间不能造成依赖循环 A -> B,B -> C,C -> A
    [operation1 addDependency:operation2];
    
    [queue addOperations:@[operation1,operation2,operation3] waitUntilFinished:NO];
}

NSOperation并发数与挂起

最大并发数

若是开启线程较多,会增大CPU工作量,从而降低了每个线程被调用的次数.为了合理的提高CPU的工作效率,我们可以设置队列最大并发数.

  • 可以通过为maxConcurrentOperationCount属性赋值即可,maxConcurrentOperationCount的默认值为-1,即不限并发数量.当其设置后队列变成串行队列.

@interface ViewController ()

@property (nonatomic, strong)NSOperationQueue *queue;

@end

//最大并发数量
- (void)maxConcurrentOperationsCount{
    _queue = [[NSOperationQueue alloc] init];
    //设置为2后表示每次只能并发执行2个任务,变成了串行执行,如果设置为1就是每次只能并发执行一个任务
    //下面的执行就是1和2先并发执行,然后3和4并发执行,入股偶设置为1则按顺序1 2 3 4一个一个执行
    _queue.maxConcurrentOperationCount = 2;
    [_queue addOperationWithBlock:^{
        for (int i = 0; i < 5000; i++) {
            NSLog(@"--->1<---%@",[NSThread currentThread]);
        }
    }];
    
    [_queue addOperationWithBlock:^{
        for (int i = 0; i < 5000; i++) {
            NSLog(@"--->2<---%@",[NSThread currentThread]);
        }
    }];
    
    [_queue addOperationWithBlock:^{
        for (int i = 0; i < 5000; i++) {
            NSLog(@"--->3<---%@",[NSThread currentThread]);
        }
    }];
    
    [_queue addOperationWithBlock:^{
        for (int i = 0; i < 5000; i++) {
            NSLog(@"--->4<---%@",[NSThread currentThread]);
        }
    }];
}

队列的挂起与取消

挂起

suspended:将其赋值为YES即挂起,赋值为NO即恢复

  • 当线程处于执行状态,设置队列挂起时不会影响其执行,受影响的是那些还没有执行的任务

  • 当队列设置为挂起状态后,可以修改其状态再次恢复任务

取消

设置取消的方法是cancelAllOperations,当取消任务后不可以恢复

若任务操作时自定义的NSOperation类型的话,执行完一个耗时操作后需要加一是否取消任务的判断,再去执行另外一个耗时操作.同样取消不影响当前正在执行的线程,后面的线程会被取消.

//挂起与取消
- (void)suspendedAndCancel{
    //挂起 当前正在执行的任务不受影响,后面还未执行的任务会被挂起,后面的任务可以被恢复
//    [self.queue setSuspended:!self.queue.suspended];
    //取消 当前所有任务会被取消,并且不可恢复
    [self.queue cancelAllOperations];
}

- (void)customOperation{
    NSOperationQueue *queue = [[NSOperationQueue alloc] init];
    PJCustomOperation *customOperation1 = [[PJCustomOperation alloc] init];
    [queue addOperation:customOperation1];
    self.queue = queue;
}
@implementation PJCustomOperation

- (void)main{
    
    //如果任务任务被取消了就不往下执行,不做这个判断的话,当队列被取消是当前的线程不受影响,故如果该线程有多个耗时操作会继续进行下去
    if(self.isCancelled){
        return;
    }
    
    //模拟在同一个线程里面做多个耗时操作
    for (int i = 0; i < 50000; i++) {
        NSLog(@"--->PJCustomOperation1<---%@",[NSThread currentThread]);
    }
    
    //如果任务任务被取消了就不往下执行,不做这个判断的话,当队列被取消是当前的线程不受影响,故如果该线程有多个耗时操作会继续进行下去
    if(self.isCancelled){
        return;
    }
    
    for (int i = 0; i < 50000; i++) {
        NSLog(@"--->PJCustomOperation2<---%@",[NSThread currentThread]);
    }
    
    //如果任务任务被取消了就不往下执行,不做这个判断的话,当队列被取消是当前的线程不受影响,故如果该线程有多个耗时操作会继续进行下去
    if(self.isCancelled){
        return;
    }
    
    for (int i = 0; i < 50000; i++) {
        NSLog(@"--->PJCustomOperation3<---%@",[NSThread currentThread]);
    }
    
    //如果任务任务被取消了就不往下执行,不做这个判断的话,当队列被取消是当前的线程不受影响,故如果该线程有多个耗时操作会继续进行下去
    if(self.isCancelled){
        return;
    }
    
    for (int i = 0; i < 50000; i++) {
        NSLog(@"--->PJCustomOperation4<---%@",[NSThread currentThread]);
    }
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,657评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,662评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,143评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,732评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,837评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,036评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,126评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,868评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,315评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,641评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,773评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,470评论 4 333
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,126评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,859评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,095评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,584评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,676评论 2 351

推荐阅读更多精彩内容

  • 学习多线程,转载两篇大神的帖子,留着以后回顾!第一篇:关于iOS多线程,你看我就够了 第二篇:GCD使用经验与技巧...
    John_LS阅读 613评论 0 3
  • 欢迎大家指出文章中需要改正或者需要补充的地方,我会及时更新,非常感谢。 一. 多线程基础 1. 进程 进程是指在系...
    xx_cc阅读 7,179评论 11 70
  • 一、前言 上一篇文章iOS多线程浅汇-原理篇中整理了一些有关多线程的基本概念。本篇博文介绍的是iOS中常用的几个多...
    nuclear阅读 2,047评论 6 18
  • 多线程 在iOS开发中为提高程序的运行效率会将比较耗时的操作放在子线程中执行,iOS系统进程默认启动一个主线程,用...
    郭豪豪阅读 2,591评论 0 4
  • 在这篇文章中,我将为你整理一下 iOS 开发中几种多线程方案,以及其使用方法和注意事项。当然也会给出几种多线程的案...
    张战威ican阅读 603评论 0 0