前言
记得去年写一个项目,有一个需求,就是按照用户在输入框内输入的文字和图片顺序进行显示,当时我只是把图片的下载放到了异步线程,但是下载下来的图片的顺序和用户上传的顺序不一样。在这里自己做个记录吧。这个demo是看的别人的但是忘了链接了,所以把自己写的链接过来了。
第一种NSOperation完成
要想有顺序的下载图片,就必须加一个 addDependency 这个方法 ,大概意思就是下一个线程必须在这个这个线程运行完成在运行。从代码中粘了一点,op1必须在op完成之后再运行。
NSOperation *op=[NSBlockOperation blockOperationWithBlock:^{NSString *str =self.urlStrs[0]; NSURL *url=[NSURL URLWithString:str]; UIImage *image=[UIImage imageWithData:[NSData dataWithContentsOfURL:url]]; [weakSelf.images addObject:image]; //刷新UI一定要在主线程 刚开始没有在主线程 显示的非常慢 [[NSOperationQueue mainQueue] addOperationWithBlock:^{ [weakSelf.tableView reloadData]; }]; }];
NSOperation *op1=[NSBlockOperation blockOperationWithBlock:^{ NSString *str =self.urlStrs[1]; NSURL *url=[NSURL URLWithString:str]; UIImage *image=[UIImage imageWithData:[NSData dataWithContentsOfURL:url]]; [weakSelf.images addObject:image]; [[NSOperationQueue mainQueue] addOperationWithBlock:^{ [weakSelf.tableView reloadData]; }]; }];
/有序的下载就是添加一个依赖 [op1 addDependency:op]; op下载完成之后再下载op1 依次类推 就是有序的了 不同的NSOperationQueue 可以添加依赖/
[op1 addDependency:op];
第二种GCD完成
其实GCD使用的方法和NSOperation类似,只是方法不同。GCD有个信号量的这么个属性,也是根据这个来限制只有当前线程完成在去执行另一个线程的方法。
- 给信号量初始化为1,只允许走一个线程
dispatch_semaphore_t sema=dispatch_semaphore_create(1);
- 信号量有一个dispatch_semaphore_wait这个方法,让信号量减一,变成零了,就不走别的线程了。
dispatch_semaphore_wait(sema, DISPATCH_TIME_FOREVER);>
- 当调用这个dispatch_semaphore_signal方法的时候,信号量就又会加一,接着走下面的线程。
dispatch_semaphore_signal(sema);
部分代码
dispatch_semaphore_t sema=dispatch_semaphore_create(1); typeof(self) weakSelf=self; _groupg=dispatch_group_create(); dispatch_semaphore_wait(sema, DISPATCH_TIME_FOREVER); dispatch_group_enter(_groupg);
dispatch_group_async(_groupg, dispatch_get_global_queue(0, 0), ^{
NSString *str =self.urlStrs[0]; NSURL *url=[NSURL URLWithString:str]; UIImage *image=[UIImage imageWithData:[NSData dataWithContentsOfURL:url]]; [weakSelf.images addObject:image]; dispatch_group_leave(_groupg); dispatch_semaphore_signal(sema); });
第三种用串行队列完成
贴出来部分代码
typeof(self) weakSelf=self; char *lable="RWN"; dispatch_queue_t seart=dispatch_queue_create(lable, DISPATCH_QUEUE_SERIAL); dispatch_async(seart, ^{ for (int i=0; i<4; i++) { NSString *str =self.urlStrs[i]; NSURL *url=[NSURL URLWithString:str]; UIImage *image=[UIImage imageWithData:[NSData dataWithContentsOfURL:url]]; [weakSelf.images addObject:image]; } dispatch_async(dispatch_get_main_queue(), ^{ [weakSelf.tableView reloadData]; }); });
这三种方法都可以实现,说一些自己的感受吧,之前一直感觉NSOperation比较神秘,感觉会比较难,自己学不会,一直抱着这个心态,所以对多线程一直不太理解,等你真正看了之后才发现,原来也没有。所以说程序员还是要多学习的,大家共同努力吧!有什么不正确的可以留言互相讨论。
一个步入中年的程序员