总结多线程随笔(一)

分为显式调用和隐式调用。
显式调用的类为NSThread.

[NSThread detachNewThreadSelector:@selector(run:) toTarget:target withObject:obj];
[NSThread detachNewThreadWithBlock:^{}];
//拿不到线程
NSThread *newThread = [[NSThread alloc] initWithTarget:target selector:@selector(run:) object:obj];
NSThread *newThread = [[NSThread alloc] init];
NSThread *newThread = [[NSThread alloc] initWithBlock:^{}];
//可以拿到这个线程
//重点:有block的都是iOS10的

可以给线程起个名字。方便一点。

@property (nullable, copy) NSString *name;
newThread.name = @"1234567890";
NSLog(@"%@", newThread);
2017-10-10 11:00:35.612 demo[1597:95527] <NSThread: 0x600000071500>{number = 3, name = 1234567890}

启动子程序。

[newThread start];
//init创建的走了这行代码newThread才会跑方法。

退出当前线程.
有两个方法。一个是类方法exit,一个是实例方法cancel。exit是立刻停止。一般写在block内部。cancel是改变线程的isCancelled。然后自己判断。上代码:

NSThread *newThread = [[NSThread alloc] initWithBlock:^{
    NSLog(@"22222222, %@", [NSThread currentThread]);
    NSLog(@"11111111, %@", [NSThread currentThread]);
}];
打印结果:
2017-10-10 11:00:35.612 demo[1597:95599] 22222222, <NSThread: 0x600000071500>{number = 3, name = 1234567890}
2017-10-10 11:00:35.612 demo[1597:95599] 11111111, <NSThread: 0x600000071500>{number = 3, name = 1234567890}
/*我是一条分割线*/
NSThread *newThread = [[NSThread alloc] initWithBlock:^{
    NSLog(@"22222222, %@", [NSThread currentThread]);
    [NSThread exit];
    NSLog(@"11111111, %@", [NSThread currentThread]);
}];
2017-10-10 11:00:35.612 demo[1597:95599] 22222222, <NSThread: 0x600000071500>{number = 3, name = 1234567890}
//第二个里面加入了退出代码。所以只打印了一次。但是这种方法,需要自己手动释放内存。否则容易有内存泄漏。一般配合自动释放池使用。

线程优先级

@property double threadPriority;
//这个属性在官方里面写着To be deprecated; use qualityOfService below,即被弃用的意思。
@property NSQualityOfService qualityOfService;
typedef NS_ENUM(NSInteger, NSQualityOfService) {
    NSQualityOfServiceUserInteractive = 0x21,
    //最高优先级。用于直接的UI交互。比如事件的处理和UI绘制。
    NSQualityOfServiceUserInitiated = 0x19,
    //次高优先级。用于进一步的UI交互。比如点击按钮后的事件。
    NSQualityOfServiceUtility = 0x11,
    //通常优先级。此工作可能已被用户请求或自动启动,不会阻止用户进一步交互,通常在用户可见的时间尺度上运行。比如网络请求,导入视频等。
    NSQualityOfServiceBackground = 0x09,
    //后台优先级。那些用户可能不知道其运行的。为其他优先级让路的操作。比如数据备份。
    NSQualityOfServiceDefault = -1
    //默认优先级。
} ;
double newThreadPriority = [NSThread threadPriority];
BOOL isSet = [NSThread setThreadPriority:1.0];
//范围[0.0, 1.0],越大优先级越高。

线程暂停

    NSThread *newThread = [[NSThread alloc] initWithBlock:^{
        while (1) {
            NSLog(@"111");
            [NSThread sleepUntilDate:[NSDate dateWithTimeIntervalSinceNow:1]];
            //休眠到指定时间点
            NSLog(@"222");
            [NSThread sleepForTimeInterval:1];
            //从此休眠多少秒
        }
    }];
2017-10-10 13:56:45.739 demo[2264:168074] 111
2017-10-10 13:56:46.742 demo[2264:168074] 222
2017-10-10 13:56:47.743 demo[2264:168074] 111
2017-10-10 13:56:48.745 demo[2264:168074] 222

栈容量

@property NSUInteger stackSize;
//默认为512 * 1024.最小设置为 8 * 1024,且一个是4的倍数 * 1024。

NSThread的类属性

@property (class, readonly, strong) NSThread *mainThread ;
@property (class, readonly) BOOL isMainThread ; 
[NSThread mainThread];
[NSThread isMainThread];
//这两个是类属性,且只读。可以看作是两个类方法。第一个获取主线程,第二个判断当前线程是否为主线程。
@property (class, readonly, copy) NSArray<NSNumber *> *callStackReturnAddresses;
@property (class, readonly, copy) NSArray<NSString *> *callStackSymbols;
//线程的调用栈里有一个个活动帧(frame)。活动帧由三部分组成:函数参数,返回地址(Return Address),帧内的变量。callStackReturnAddresses就是这些frame的返回地址。callStackSymbols获取当前的调用栈。

隐式调用
通过NSObject的Category方法调用

  • (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;
  • (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;
    //SEL是要执行的方法,wait是是否等待当前线程执行完毕再执行,YES是等待,withObject当SEL需要传参时的参数可以为nil,modes为此线程的runloop modes。
  • (void)performSelectorInBackground:(SEL)aSelector withObject:(nullable id)arg ;
    //单纯的生成一个新线程并执行方法

希望大家多提宝贵意见,谢谢

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 开启线程 分离主线程创建:创建线程后会自动执行,但是线程外部不可获取到该线程对象detachNewThreadWi...
    Mr_Pt阅读 1,104评论 0 1
  • 一、多线程简介: 所谓多线程是指一个 进程 -- process(可以理解为系统中正在运行的一个应用程序)中可以开...
    寻形觅影阅读 1,128评论 0 6
  • 1.术语线程:用于指代独立执行的代码段进程:用于指代一个正在运行的可执行程序,它可以包含多个线程。任务:用于指代抽...
    Jason_KB阅读 222评论 0 0
  • 在IOS多线程中NSThread这种方式虽然简单易用,但是需要程序员管理,所以在实际开发中使用较少。这里只简单介绍...
    天外丶飞仙阅读 175评论 0 0
  • 因为什么,我最新重新学习了多线程的知识,顺便总结一下。 首先,为什么要使用多线程,在iOS中我们希望主线程可以快速...
    yezi1989阅读 242评论 0 1