iOS开发多线程篇-NSThread

上篇我们学习了iOS多线程解决方式中的NSOperation,这篇我主要概况总结iOS多线程中NSThread的解决方式和基本用例

一.iOS多线程对比

  1. NSThread
    每个NSThread对象对应一个线程,真正最原始的线程

    • 优点:NSThread轻量级最轻,相对简单
    • 缺点:手动管理所有的线程活动,如生命周期、线程同步、睡眠等
  2. NSOperation
    自带线程管理的抽象类

    • 优点:自带线程周期管理,操作上可更注重自己逻辑
    • 缺点:面向对象的抽象类,只能实现它或者使用它定义好的两个子类:NSInvocationOperationNSBlockOperation
  3. GCD
    Grand Central Dispatch 是Apple开发的一个多核编程的解决方法。

    • 优点:最高效,避开并发陷阱
    • 缺点:基于C实现
  4. 选择小结

    • 简单而安全的选择NSOperation实现多线程即可
    • 处理大量并发数据,又追求性能效率的选择GCD
    • NSThread较麻烦,不建议使用

二. 场景选择

  1. 图片异步加载:这种常见的场景是最常见也是必不可少的,异步加载图片又分成两种

    • 在UI主线程开启新线程按顺序加载图片,加载完成刷新UI
    • 依然是在主线程开启新线程,顺序不定的加载图片,加载完成后刷新UI
  2. 创作工具上的异步:这个跟上边任务调度道理差不多,只是为了丰富描述,有助于“举一反三”效果,如下描述的是APP创作小说

    • app本地创作10个章节内容完成同步服务器,接着同时发表这10个章节将产生的一系列动作,其中上传内容,获取分配章节Id,如果后台没有做处理最好方式做异步按顺序执行。
    • app本地创作列表中有3本小说要发表,如果同时发表创作列表中的3本小说,自然考虑并行队列执行发表。

三.使用方法

  1. 三种实现开启线程方式:

    • 动态实例化
    NSThread * thread = [[NSThread alloc] initWithTarget:self selector:@selector(loadImageSource:) object:imgUrl];
    thread.threadPriority = 1;  //设置线程的优先级(0.0 - 1.0, 1.0最高级)
    [thread start];
  • 静态实例化

    [NSThread detachNewThreadSelector:@selector(loadImageSource:) toTarget:self withObject:imgUrl];
  • 隐式实例化
    [self performSelectorOnMainThread:@selector(loadImageSource:) withObject:self waitUntilDone:imgUrl];

代码示例:

//
//  ViewController.m
//  TestNSThread
//
//  Created by taobaichi on 2017/3/21.
//  Copyright © 2017年 MaChao. All rights reserved.
//

#import "ViewController.h"
#define imgUrl @"http://avatar.csdn.net/2/C/D/1_totogo2010.jpg"

@interface ViewController ()

@property (nonatomic, strong) UIImageView * imageView;

@end

@implementation ViewController

- (void)viewDidLoad {
    [super viewDidLoad];

    self.imageView = [[UIImageView alloc]initWithFrame:CGRectMake(self.view.frame.size.width /2 - 100, self.view.frame.size.height / 2 - 100, 200, 200)];
    self.imageView.backgroundColor = [UIColor purpleColor];
    [self.view addSubview:self.imageView];

    [self dynamicCreateThread];
}

//动态创建线程
-(void)dynamicCreateThread{
    NSThread * thread = [[NSThread alloc] initWithTarget:self selector:@selector(loadImageSource:) object:imgUrl];
    thread.threadPriority = 1;  //设置线程的优先级(0.0 - 1.0 1.0最高级)
    [thread start];
}

//静态创建线程
-(void)staticCreateThread{
    [NSThread detachNewThreadSelector:@selector(loadImageSource:) toTarget:self withObject:imgUrl];
}

//隐式创建线程
-(void)implicitCreateThread{
    [self performSelectorInBackground:@selector(loadImageSource:) withObject:imgUrl];
}

-(void)loadImageSource:(NSString *)url
{
    NSData * imgData = [NSData dataWithContentsOfURL:[NSURL URLWithString:url]];
    UIImage * image = [UIImage imageWithData:imgData];

    if (imgData != nil) {
        //回到主线程刷新UI
        [self performSelectorOnMainThread:@selector(refreshImageView:) withObject:image waitUntilDone:YES];
    }else{
        NSLog(@"there no image data");
    }
}

-(void)refreshImageView:(UIImage *)image{

    [self.imageView setImage:image];
}

@end

2.NSThread的拓展认识

  • 获取当前线程
NSThread * current = [NSThread currentThread];
  • 获取主线程
NSThread * main = [NSThread mainThread];
  • 暂停当前线程
[NSThread sleepForTimeInterval:2.0];
  • 线程之间通信
    //在指定线程上执行
    [self performSelector:@selector(refreshImageView:) onThread:thread withObject:image waitUntilDone:YES];

    //在主线程执行
    [self performSelectorOnMainThread:@selector(refreshImageView:) withObject:image waitUntilDone:YES];

    //在后台执行
    [self performSelectorInBackground:@selector(refreshImageView:) withObject:image];

    //在当前 线程上执行
    [self performSelector:@selector(refreshImageView:) withObject:image];
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值