使用多线程加载网络图片

NSThread
1、两种方式:
。手动开启方式

/*
     * 创建手动开启方式
     *第三个参数:就是方法选择器选择方法的参数
     */
 NSThread *thread = [[NSThread alloc]initWithTarget:self selector:@selector(thread) object:@"thread"];
//    开启线程
  [thread start];

。自动开启方式

[NSThread detachNewThreadSelector:@selector(thread1:) toTarget:self withObject:@"thread1"];

2、加载一张图片的步骤:

1、创建一个UIImageView,并放在父视图上
2、创建一个子线程
3、通过url获取网络图片
4、回到主线程
5、在主线程更新UI

图片地址的宏定义:

#define kurl @"http://store.storeimages.cdn-apple.com/8748/as-images.apple.com/is/image/AppleInc/aos/published/images/s/38/s38ga/rdgd/s38ga-rdgd-sel-201601?wid=848&hei=848&fmt=jpeg&qlt=80&op_sharpen=0&resMode=bicub&op_usm=0.5,0.5,0,0&iccEmbed=0&layer=comp&.v=1454777389943"

代码如下:

- (void)viewDidLoad {
    [super viewDidLoad];
[NSThread detachNewThreadSelector:@selector(thread1:) toTarget:self withObject:@"thread1"];
imageView = [[UIImageView alloc]initWithFrame:CGRectMake(50, 50, 200, 200)];
    [self.view addSubview:imageView];
}
//在子线程执行的方法
- (void)thread1:(NSString *)sender{

   [NSThread currentThread];//获取到当前所在的信息
    NSThread *thread = [NSThread currentThread];
    thread.name = @"我是子线程 ";
 NSLog(@"%@",thread);   
   [NSThread isMainThread] // 判断当前线程是否是主线程
  BOOL isMainThread = [NSThread isMainThread];   
  [NSThread isMultiThreaded] //判断是否是多线程
    BOOL isMUltiThread = [NSThread isMultiThreaded];   
   NSLog(@"%d,%d",isMainThread,isMUltiThread);  
//  设置线程的优先级(0-1) setThreadPriority:
  [NSThread setThreadPriority:1.0];
 // sleepForTimeInterval:让线程休眠
   [NSThread sleepForTimeInterval:2];
    
//    从网络加载图片并将它转化为data类型的数据
    NSData *data = [NSData dataWithContentsOfURL:[NSURL URLWithString:kUrl]];
    image = [UIImage imageWithData:data];
  
//    waiUntilDone设为YES,意味着UI更新完才会做其它操作
    [self performSelectorOnMainThread:@selector(updateUI:) withObject:image waitUntilDone:YES];
    
}

- (void)updateUI:(UIImage *)kimage{ 
https://github.com/duiyueliu/iOS-Images-Extractor-master.git
    imageView.image = kimage;
    NSLog(@"updateUI方法所在的线程%@",[NSThread currentThread]);
}

3、加载多张图片跟加载一张图片的步骤一样,就直接上代码了:

- (void)viewDidLoad {
    [super viewDidLoad];
 imageIndex = 100;
    threadArrays = [NSMutableArray array];
//    创建多个UIImageView
    
    for (int row = 0; row<3; row++) {
        for (int list = 0; list<2; list++) {
            
            UIImageView *imageView = [[UIImageView alloc]initWithFrame:CGRectMake(5+list*205, 5+row*205, 200, 200)];
            imageView.backgroundColor = [UIColor grayColor];
            imageView.tag = imageIndex++;
            [self.view addSubview:imageView];
            
        }
    }
    
//    创建多个子线程
    
    for (int index = 0; index<6; index++) {
//        [NSThread detachNewThreadSelector:@selector(thread:) toTarget:self withObject:@(index)];
        NSThread *thread = [[NSThread alloc]initWithTarget:self selector:@selector(thread:) object:@(index)];
        [thread start];
        [threadArrays addObject:thread];
    }
    
    
    
}
//通过url加载网络图片
- (void)thread:(NSNumber *)index{
   
    //    通过线程休眠 实现   实现图片的顺序加载
    [NSThread sleepForTimeInterval:[index intValue]];
    
    NSThread *thread = [NSThread currentThread];
    
    if (thread.isCancelled == YES) {
        [NSThread exit];
    }
    
       NSData *data = [NSData dataWithContentsOfURL:[NSURL URLWithString:kUrl]];
    image = [UIImage imageWithData:data];
    
//     回到主线程
    [self performSelectorOnMainThread:@selector(updateUI:) withObject:index  waitUntilDone:YES];
    
}
//
- (void)updateUI:(NSNumber *)index{

    UIImageView *imageView = [self.view viewWithTag:[index intValue] + 100];
    imageView.image = image;
}

- (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event{

    for (int index = 0; index<6; index++) {
        
        NSThread *thread = threadArrays[index];
        if (thread.isFinished == NO) {
//            点击屏幕  取消未完成的线程
            [thread cancel];
        }
    }
    
    [NSThread currentThread];
    NSLog(@"%@",threadArrays);
    
}

NSOperation

介绍:采用NSOperation(线程操作,通常用它的子类)和NSOperationQueue(线程队列)搭配来做多线程开发,采用NSOperation指定一个操作,把这个操作放到线程队列(线程池)中,让线程队列安排它的周期。

1、加载一张图片的步骤:

1、创建视图
2、创建线程操作
3、创建线程队列
4、把线程操作放在线程队列中
5、在子线程加载网络资源
6、回到主线程
7、在主线程更新UI

2、三种方式:

方式一:NSInvocationOperation和NSOperationQueue搭配进行多线程开发

图片地址的宏定义:

#define kurl @"http://store.storeimages.cdn-apple.com/8748/as-images.apple.com/is/image/AppleInc/aos/published/images/s/38/s38ga/rdgd/s38ga-rdgd-sel-201601?wid=848&hei=848&fmt=jpeg&qlt=80&op_sharpen=0&resMode=bicub&op_usm=0.5,0.5,0,0&iccEmbed=0&layer=comp&.v=1454777389943"
//     1、创建视图
    imageView = [[UIImageView alloc]initWithFrame:CGRectMake(50, 50, 200, 200)];
    [self.view addSubview:imageView];
//    2、创建线程操作
    NSInvocationOperation *invocationOperation = [[NSInvocationOperation alloc]initWithTarget:self selector:@selector(loadResource) object:nil];
//    3、创建线程队列
    NSOperationQueue *operationQueue = [NSOperationQueue new];
//    4、把线程操作放在线程队列中
    [operationQueue addOperation:invocationOperation];
}
//5、在子线程加载网络资源
- (void)loadResource{
    NSData *data = [NSData dataWithContentsOfURL:[NSURL URLWithString:kurl]];
    UIImage *image = [UIImage imageWithData:data];
    
//    6、回到主线程
    
    [[NSOperationQueue mainQueue]addOperationWithBlock:^{
        
//         7、在主线程更新UI
        imageView.image = image;
        
    }];
}

方式二:NSBlockOperation和NSOperationQueue搭配

- (void)viewDidLoad{
    [super viewDidLoad];

    
//     1、创建视图
    imageView = [[UIImageView alloc]initWithFrame:CGRectMake(50, 50, 200, 200)];
        [self.view addSubview:imageView];
    
//    2、创建一个线程操作
    NSBlockOperation *blockOperation = [NSBlockOperation blockOperationWithBlock:^{
        
//        5、加载网络资源
        NSData *data = [NSData dataWithContentsOfURL:[NSURL URLWithString:kurl]];
        UIImage *image = [UIImage imageWithData:data];
        
//        6、回到主线程
        [[NSOperationQueue mainQueue]addOperationWithBlock:^{
//            7、更新UI
            imageView.image = image;
        }];
        
    }];
    
//    3、创建一个线程队列
    NSOperationQueue *operationQueue = [NSOperationQueue new];
    
//    4、把线程操作放到线程队列中
    [operationQueue addOperation:blockOperation];
 
}

方式三:用自定义于NSOperation的类与NSOperationQueue搭配

- (void)viewDidLoad{

    [super viewDidLoad];
    self.view.backgroundColor = [UIColor grayColor];
//     1、创建视图
        imageView = [[UIImageView alloc]initWithFrame:CGRectMake(50, 50, 200, 200)];
        [self.view addSubview:imageView];
    
//    2、创建一个线程操作,在类中重写main方法,在main指定要进行的操作
    CusTomOperation *customOperation = [[CusTomOperation alloc]initWithImageView:imageView];
    
//    3、创建一个线程队列
    NSOperationQueue *operationQueue = [NSOperationQueue new];
    
//    4、将线程操作放到线程队列中
    [operationQueue addOperation:customOperation];
    
    
}

自定义的NSOperation类中.m文件中的代码如下:

- (instancetype)initWithImageView:(UIImageView *)imageView
{
    self = [super init];
    if (self) {
        _imageView = imageView;
    }
    return self;
}
- (void)main{

//    自动创建一个自动释放池,因为在这里无法访问到主线程的自动释放池
    @autoreleasepool {
        
//        5、在子线程加载网络资源
        NSData *data = [NSData dataWithContentsOfURL:[NSURL URLWithString:kurl]];
        UIImage *image = [UIImage imageWithData:data];
        
//        6、回到主线程
        [[NSOperationQueue mainQueue]addOperationWithBlock:^{
          
//         7、更新UI
           _imageView.image = image;
            
        }];
    }
}
@end

第三种方式使用时需要注意的两点是:
(1)该子类需重写main方法,在main方法内做线程操作,该线程被执行就会自动调用main方法
(2)在main方法内切记要新建一个自动释放池,因为如果是同步操作,该方法能够自动访问到主线程的自动释放池,如果是异步执行操作,那么将无法访问到主线程的自动释放池。

下面用第二种方式加载多张图片,步骤跟加载一张图片的步骤一样,代码如下:

#define kurl @"http://store.storeimages.cdn-apple.com/8748/as-images.apple.com/is/image/AppleInc/aos/published/images/s/38/s38ga/rdgd/s38ga-rdgd-sel-201601?wid=848&hei=848&fmt=jpeg&qlt=80&op_sharpen=0&resMode=bicub&op_usm=0.5,0.5,0,0&iccEmbed=0&layer=comp&.v=1454777389943"
@interface MoreImageViewViewController ()
{
    int imageIndex;
    NSOperationQueue *operationQueue;
}
@end

@implementation MoreImageViewViewController


- (void)viewDidLoad{

    [super viewDidLoad];
imageIndex = 100;
//    1、创建多个视图
    for (int row = 0; row<3; row++) {
        for (int list = 0; list <2; list++) {
            UIImageView *imageView = [[UIImageView alloc]initWithFrame:CGRectMake(5+list*205, 5+row*205, 200, 200)];
            imageView.backgroundColor = [UIColor yellowColor];
            imageView.tag = imageIndex++;
            
            [self.view addSubview:imageView];
        }
    }
    
//    3、创建线程队列
    operationQueue = [NSOperationQueue new];
    
//    2、创建多个线程
    for (int index = 0; index<6; index++) {
        NSBlockOperation *blockOperation = [NSBlockOperation blockOperationWithBlock:^{
            
//            5、在子线程中加载网络资源
            NSData *data = [NSData dataWithContentsOfURL:[NSURL URLWithString:kurl]];
            UIImage *image = [UIImage imageWithData:data];
            
            [NSThread sleepForTimeInterval:0.5];
//            6、回到主线程
            [[NSOperationQueue mainQueue]addOperationWithBlock:^{
                
//               7、更新UI
                UIImageView *imageView = [self.view viewWithTag:index+100];
                imageView.image = image;
            }];
            
        }];
        
        
//        让第一个线程谁都不依赖
        if (index != 0) {
            
            [blockOperation addDependency:operationQueue.operations[index-1]];
        }
        
        
        
        //   4、将线程操作放到线程队列中
        [operationQueue addOperation:blockOperation];
  }

GCD

1、介绍:全称是Grand Central Dispath ,纯C语言编写,提供非常多强大的函数,是目前苹果官网推荐的多线程开发方法,NSOperation便是基于GCD的封装。
2、优势:
(1)为多核的并行运算提出了解决方案
(2)GCD会自动利用更多的CPU内核,比如双核,四核
(3)GCD自动管理线程的生命周期(创建线程,调度任务,销毁线程)
(4)程序员只需告诉GCD想要执行什么任务,不需要编写任何线程管理代码
3、GCD中有两个核心概念
(1)任务:执行什么操作
(2)队列:用来存放任务

4、队列可以分为两大类型
(1)串行队列(serial Dispatch Queue):只有一个线程,加入到队列中的操作按添加顺序依次执行一个任务助兴完毕后,才能执行下一个任务
(2)并发队列(Concurrent Dispatch Queue:可以有多个线程,操作进来以后它会将这些线程安排在可用的处理器上,同时保证先进来的任务优先处理
(3)还有一个特殊的队列就是主队列,主队列中永远只有一个线程——主线程,用来执行主线程的操作任务。
5、采用GCD做多线程,可以抽象为两步
(1)找到队列(主队列或串行队列或并行队列)
(2)在队列中用同步或者异步的方式执行任务
6、执行队列中任务的两种方式
(1)同步:只能在当前线程执行任务,不具备开启新线程的能力
(2)异步:可以在新的线程中执行任务,具备开启新线程的能力
7、GCD创建的线程的四种执行方式
(1)串行同步

    1、找到队列
    /*
     *第一个参数:该队列的名字
     *第二个参数:指定队列的类型
     */
    dispatch_queue_t serialQueue = dispatch_queue_create("serialQueue", DISPATCH_QUEUE_SERIAL);
    
    给队列指定任务
    /*
     *asyn是异步   syn是同步
     
     *第一个参数:任务在哪个队列中执行
     *第二个参数:想要执行的而操作
     */
    dispatch_sync(serialQueue, ^{
    
        NSLog(@"1===%@",[NSThread currentThread]);
    });

(2)串行异步

    
    1、找到队列
    dispatch_queue_t serialQueue1 = dispatch_queue_create("serialQueue1", DISPATCH_QUEUE_SERIAL);
    
    2、给队列指定异步任务
    dispatch_async(serialQueue1, ^{
       
        NSLog(@"1 = %@",[NSThread currentThread]);
    });

(3)并行同步

    1、找到一个队列
    dispatch_queue_t concurrentQueue = dispatch_queue_create("concurrentQueue", DISPATCH_QUEUE_CONCURRENT);
    
    2、给队列指定任务
    dispatch_sync(concurrentQueue, ^{
      
        NSLog(@"%@",[NSThread currentThread]);
    });

(4)并行异步

//    1、创建队列
    
    dispatch_queue_t concurrentQueue1 = dispatch_queue_create("concurrentQueue", DISPATCH_QUEUE_CONCURRENT);
//    2、给队列指定任务
    dispatch_async(concurrentQueue1, ^{
        
        NSLog(@"%@",[NSThread currentThread]);
    });

8、加载一张图片的步骤:

1、创建视图
2、创建一个串行队列
3、用异步方式执行串行队列中的任务
4、加载网络资源
5、回到主线程
6、更新UI

以异步串行方式为例加载一张图片:

#import "OneImageViewController.h"
#define kurl @"http://store.storeimages.cdn-apple.com/8748/as-images.apple.com/is/image/AppleInc/aos/published/images/s/38/s38ga/rdgd/s38ga-rdgd-sel-201601?wid=848&hei=848&fmt=jpeg&qlt=80&op_sharpen=0&resMode=bicub&op_usm=0.5,0.5,0,0&iccEmbed=0&layer=comp&.v=1454777389943"
@interface OneImageViewController ()
{
    UIImageView *imageView;
}
@end

@implementation OneImageViewController

- (void)viewDidLoad {
    [super viewDidLoad];

    imageView = [[UIImageView alloc]initWithFrame:CGRectMake(50, 50, 200, 200)];
    [self.view addSubview:imageView];
    
//    2、串行队列
    
    dispatch_queue_t serialQueue = dispatch_queue_create("serialQueue", DISPATCH_QUEUE_SERIAL);
    
//    3、用异步方式执行串行队列中的任务
    
    dispatch_async(serialQueue, ^{
   
//     4、加载网络资源
        NSData *data = [NSData dataWithContentsOfURL:[NSURL URLWithString:kurl]];
        UIImage *image = [UIImage imageWithData:data];
        
//      5、回到主线程
//       dispatch_get_main_queue()这个函数找到主队列
        dispatch_queue_t mainQueue = dispatch_get_main_queue();
        
        dispatch_sync(mainQueue, ^{
//          6、更新UI
            imageView.image = image;
            
        });
    });  
}

9、用并行方式加载多张图片:

#define kurl @"http://store.storeimages.cdn-apple.com/8748/as-images.apple.com/is/image/AppleInc/aos/published/images/s/38/s38ga/rdgd/s38ga-rdgd-sel-201601?wid=848&hei=848&fmt=jpeg&qlt=80&op_sharpen=0&resMode=bicub&op_usm=0.5,0.5,0,0&iccEmbed=0&layer=comp&.v=1454777389943"

@interface MoreImageViewViewController ()
{
    int imageIndex;
    dispatch_queue_t concurrentQueue;
}
@end

@implementation MoreImageViewViewController

- (void)viewDidLoad {
    [super viewDidLoad];
//   1、创建多个视图
    imageIndex = 100;
    for (int row = 0; row<3; row++) {
        for (int list = 0; list<2; list++) {
            
            UIImageView *imageView = [[UIImageView alloc]initWithFrame:CGRectMake(5+list*205, 5+row*205, 200, 200)];
            imageView.backgroundColor = [UIColor grayColor];
            imageView.tag = imageIndex++;
            [self.view addSubview:imageView];
            
        }
    }

//    2、找到并行队列
    
    /*
     *dispatch_get_global_queue 获取到系统的全局并行队列
     *第一个参数:是优先级
     *第二个参数:保留参数,没用
     */
//    dispatch_queue_t concurrentQueue = dispatch_get_global_queue(0, 0);
    concurrentQueue = dispatch_queue_create("concurrentQueue", DISPATCH_QUEUE_SERIAL);
    
//    3、给这个并行队列指定多个任务
    for (int index = 0; index<6; index++) {
        dispatch_async(concurrentQueue, ^{
            
//       在子线程加载网络资源
            NSData *data = [NSData dataWithContentsOfURL:[NSURL URLWithString:kurl]];
            UIImage *image = [UIImage imageWithData:data];
            
//            5、回到主线程
            dispatch_queue_t mainQueue = dispatch_get_main_queue();
            dispatch_sync(mainQueue, ^{
                
//                6、更新UI
                UIImageView *imageView = [self.view viewWithTag:100+index];
                imageView.image = image;
            });
        });
    }
    
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,837评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,551评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,417评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,448评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,524评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,554评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,569评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,316评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,766评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,077评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,240评论 1 343
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,912评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,560评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,176评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,425评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,114评论 2 366
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,114评论 2 352

推荐阅读更多精彩内容