iOS多线程之NSThread

简介

苹果对 Pthreads 进行了面向对象的封装 NSTherad,使用起来比 pthread 更加面向对象,简单易用,可以直接操作线程对象。不过也需要需要程序员自己管理线程的生命周期(主要是创建),我们在开发的过程中偶尔使用 NSThread。比如我们会经常调用[NSThread currentThread]来显示当前的进程信息。

使用

创建和启动线程

直接使用是- (instancetype)init;没有意义的
正确的创建方式

  1. 先创建线程,再启动线程
    NSThread *thread = [[NSThread alloc]initWithBlock:^{
        NSLog(@"%@", [NSThread currentThread]);
    }];
    [thread start];
    
    NSThread *thread2 = [[NSThread alloc]initWithTarget:self selector:@selector(run) object:nil];
    [thread2 start];
    
    - (void)run
    {
    NSLog(@"%@", [NSThread currentThread]);
    }
    
    
    2019-08-06 14:26:34.310385+0800 MasonryTest[5542:512035] <NSThread: 0x600001b3b480>{number = 3, name = (null)}
    2019-08-06 14:26:34.310405+0800 MasonryTest[5542:512036] <NSThread: 0x600001b3b240>{number = 4, name = (null)}
  1. 创建线程后自动启动线程
    还有其他两种创建方法 为类方法  此时不需要启动线程 同时不需要自己去管理对象的生命周期
    [NSThread detachNewThreadWithBlock:^{
        NSLog(@"%@", [NSThread currentThread]);
    }];
    
    [NSThread detachNewThreadSelector:@selector(run) toTarget:self withObject:nil];
    
    - (void)run
    {
    NSLog(@"%@", [NSThread currentThread]);
    }
    
    2019-08-06 14:30:08.407044+0800 MasonryTest[5665:522630] <NSThread: 0x600003340880>{number = 4, name = (null)}
    2019-08-06 14:30:08.407047+0800 MasonryTest[5665:522632] <NSThread: 0x600003340a00>{number = 5, name = (null)}
  1. 隐式创建并启动线程
    [self performSelectorInBackground:@selector(run) withObject:nil];
    
    - (void)run {
     NSLog(@"%@", [NSThread currentThread]);
    }

线程的相关方法

    // 获得主线程
    + (NSThread *)mainThread;    

    // 判断是否为主线程(对象方法)
    - (BOOL)isMainThread;

    // 判断是否为主线程(类方法)
    + (BOOL)isMainThread;    

    // 获得当前线程
    NSThread *current = [NSThread currentThread];

    // 线程的名字——setter方法
    - (void)setName:(NSString *)n;    

    // 线程的名字——getter方法
    - (NSString *)name;

线程状态控制方法

  1. 启动线程方法
    - (void)start;
  1. 暂停,休眠线程方法
    在线程的运行期间我们可以调用如下方法使执行此行代码的线程进入休眠。调用的时候都需要指定休眠的时间,sleepUntilDate指定的是休眠到某个时间,sleepForTimeInterval指定的是休眠的秒数。线程休眠时 Runloop 不会被事件唤醒。  
    + (void)sleepUntilDate:(NSDate *)date;
    + (void)sleepForTimeInterval:(NSTimeInterval)ti;
  1. 退出,停止线程
    - (void)cancel;
    + (void)exit;

派发任务

// 在主线程同步或异步执行任务
- (void)performSelectorOnMainThread:(SEL)aSelector withObject:(nullable id)arg waitUntilDone:(BOOL)wait modes:(nullable NSArray<NSString *> *)array;
- (void)performSelectorOnMainThread:(SEL)aSelector withObject:(nullable id)arg waitUntilDone:(BOOL)wait;

这两个方法它们还可以指定携带的参数以及是否阻塞当前线程(派发此任务的线程),还可以派发到指定的 Runloop mode。

// 在指定线程和指定 Runloop Mode 同步或异步执行任务
- (void)performSelector:(SEL)aSelector onThread:(NSThread *)thr withObject:(nullable id)arg waitUntilDone:(BOOL)wait modes:(nullable NSArray<NSString *> *)array ;
- (void)performSelector:(SEL)aSelector onThread:(NSThread *)thr withObject:(nullable id)arg waitUntilDone:(BOOL)wait ;

指定线程,指定 Runloop Mode

// 在后台线程执行任务
- (void)performSelectorInBackground:(SEL)aSelector withObject:(nullable id)arg ;

我们也可以使用performSelectorInBackground:~向系统默认的后台线程派发任务。相比之下我们不需要自己管理子线程的生命周期,省去了需要不必要的麻烦。

优先级

@property NSQualityOfService qualityOfService;

typedef NS_ENUM(NSInteger, NSQualityOfService) {
    NSQualityOfServiceUserInteractive = 0x21,
    NSQualityOfServiceUserInitiated = 0x19,
    NSQualityOfServiceUtility = 0x11,
    NSQualityOfServiceBackground = 0x09,
    NSQualityOfServiceDefault = -1
};

苹果对于 QoS 五种优先级的使用有如下建议。开发者应该严格遵守这样的规则。

  • NSQualityOfServiceUserInteractive:用来处理用户操作,例如界面刷新、动画等。优先级最高,即时执行。
  • NSQualityOfServiceUserInitiated:处理初始化任务,为将来的用户操作作准备。例如加载文件或 Email 等。基本即时执行,最多几秒延迟。
  • NSQualityOfServiceUtility:用户不需要立即结果的操作,一般伴随进度条。例如下载、数据导入、周期性的内容更新等。几秒到几分钟延迟。
  • NSQualityOfServiceBackground:用于用户不可见的操作。例如简历索引、预加载、同步等。几分钟到数小时延迟。
  • NSQualityOfServiceDefault:默认的 QoS 用来表示缺省值。当有可能通过其它途径推断出可能的 QoS 信息时,则使用推断出的 Qos。如果不能推断,则使用 UserInitiated 和 Utility 之间的 QoS。
    Utility 及以下的优先级会受到 iOS9 中低电量模式的控制。另外,在没有用户操作时,90% 任务的优先级都应该在 Utility 之下。

线程状态信息

线程在运行期间有多种状态。可以用如下三个属性来判断线程的运行状态。
@property (readonly, getter=isExecuting) BOOL executing; // 正在运行
@property (readonly, getter=isFinished) BOOL finished; // 已经完成
@property (readonly, getter=isCancelled) BOOL cancelled; // 已经取消
©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

相关阅读更多精彩内容

  • 什么是线程?什么是进程?线程和进程什么关系?这不是我们这一章关心的重点,我在这里假设大家都知道线程是cpu调度的最...
    随风流逝阅读 2,990评论 0 1
  • 1、简介:1.1 iOS有三种多线程编程的技术,分别是:1.、NSThread2、Cocoa NSOperatio...
    LuckTime阅读 5,211评论 0 1
  • iOS多线程开发基础概念 进程 VS 线程 进程:程序的一次执行,是正在执行的程序的实例,它是Unix的一个基本概...
    qingmarch阅读 3,056评论 0 1
  • NSThread 特点:更加面向对象简单易用,可直接操作线程对象使用语言:OC语言使用频率:偶尔使用线程生命周期:...
    BWLi420阅读 3,630评论 0 1
  • 国庆之后真的很糟糕,有时候在深夜偷偷的问自己是不是不适合写代码?是不是不适合当程序员?是不是要转行才能见到未来?是...
    土鳖不土阅读 3,833评论 0 0

友情链接更多精彩内容