iOS: NSObject中执行Selector的相关方法

本文转载至 http://www.mgenware.com/blog/?p=463

1. 对当前Run Loop中Selector Sources的取消

NSObject中的performSelector:withObject:afterDelay:方法将会在当前线程的Run Loop中根据afterDelay参数创建一个Timer,如果没有调用有inModes参数的方法,该Timer会运行在当前Run Loop的默认模式中,也就是NSDefaultRunLoopMode定义的模式中。

performSelector:withObject:afterDelay:方法的使用看起来还是很简单的。这里讲另外一个辅助函数,NSObject中静态的cancelPreviousPerformRequestsWithTarget方法。该方法就是专门用来取消取消performSelector:withObject:afterDelay:方法所创建的Selector source(内部上就是一个Run Loop的Timer source)。因此该方法和performSelector:withObject:afterDelay:方法一样,只限于当前Run Loop中。

我们可以利用cancelPreviousPerformRequestsWithTarget直接取消一个对象在当前Run Loop中的所有未执行的performSelector:withObject:afterDelay:方法所产生的Selector Sources,如下代码:

- (void)viewDidLoad{    [super viewDidLoad];

    [self performSelector:@selector(test:) withObject:nil afterDelay:1];    [self performSelector:@selector(test:) withObject:@"mgen" afterDelay:2];    [NSObject cancelPreviousPerformRequestsWithTarget:self];    }

- (void)test:(id)obj{    NSLog(@"调用成功: %@", obj);}

不会有任何输出,因为两个调用都被取消了。

如果想取消单独一个的话,需使用cancelPreviousPerformRequestsWithTarget:selector:object:方法,注意selector和object参数需要一一对应。如下代码:

- (void)viewDidLoad{    [super viewDidLoad];

    [self performSelector:@selector(test:) withObject:[NSNumber numberWithInt:26] afterDelay:1];    [self performSelector:@selector(test:) withObject:[NSNumber numberWithInt:17] afterDelay:2];    [self performSelector:@selector(test:) withObject:[NSNumber numberWithInt:17] afterDelay:3];    [NSObject cancelPreviousPerformRequestsWithTarget:self selector:@selector(test:) object:[NSNumber numberWithInt:17]];    }

- (void)test:(id)obj{    NSLog(@"调用成功: %@", obj);}

只会输出:

调用成功: 26

其他两个Selector都被取消了。

返回目录

2. 在NSThread中执行Selector

这个话题很简单,直接通过NSObject的performSelectorInBackground:withObject:方法就可以,如下代码:

- (void)viewDidLoad{    [super viewDidLoad];

    [self threadInfo:@"UI"];    [self performSelectorInBackground:@selector(test:) withObject:nil];

}

- (void)test:(id)obj{    @autoreleasepool    {        [self threadInfo:@"test"];    }}- (void)threadInfo:(NSString*)category{    NSLog(@"%@ - %@", category, [NSThread currentThread]);}

输出:

UI - <NSThread: 0x71639e0>{name = (null), num = 1}test - <NSThread: 0x7176ad0>{name = (null), num = 3}

这个方法完全等效于NSThread的detachNewThreadSelector:toTarget:withObject:静态方法,那么上面NSObject的performSelectorInBackground:withObject:方法调用完全可以替换成:

[NSThread detachNewThreadSelector:@selector(test:) toTarget:self withObject:nil];

当然,用户也可以自行手动创建一个NSThread来完成上述功能,代码如下:

NSThread *thread = [[NSThread alloc] initWithTarget:self selector:@selector(test:) object:nil];[thread start];

这两种方法运行后的输出是和第一种类似的。

返回目录

3. 在NSThread中的Run Loop中执行Selector

这里需要的方法是NSObject的performSelector:onThread:withObject:waitUntilDone:方法。由于是在另一个NSThread中执行Selector,所以我们需要手动开始Run Loop。首先需要在ViewController中定义两个字段,分别是NSThread和控制线程内Run Loop执行的flag。

@interface ViewController (){    NSThread *_thread;    BOOL _isNewThreadAborted;}

接下来做的是执行这个线程,并且在线程中手动调用NSRunLoop的runMode:beforeDate:方法。这里注意,如果Run Loop没有任何Source的话,该方法会立即返回,所以需要创建一个循环来持续调用Run Loop的runMode:beforeDate:方法。并在Selector执行结束后同时尝试结束这个循环。最终代码如下:

- (void)viewDidLoad{    [super viewDidLoad];

    [self threadInfo:@"UI"];

    _isNewThreadAborted = NO;    _thread = [[NSThread alloc] initWithTarget:self selector:@selector(newThread:) object:nil];    //开始线程    [_thread start];    //在另一个线程中的Run Loop中执行Selector    [self performSelector:@selector(test:) onThread:_thread withObject:nil waitUntilDone:NO];    }

//在新线程中创建并开始一个NSRunLoop- (void)newThread:(id)obj{    @autoreleasepool    {        NSRunLoop *currentRunLoop = [NSRunLoop currentRunLoop];        while (!_isNewThreadAborted)        {            [currentRunLoop runMode:NSDefaultRunLoopMode beforeDate:[NSDate distantFuture]];        }        NSLog(@"线程停止");    }}

//Selector执行- (void)test:(id)obj{    [self threadInfo:@"test"];    _isNewThreadAborted = YES;}

- (void)threadInfo:(NSString*)category{    NSLog(@"%@ - %@", category, [NSThread currentThread]);}

输出:

UI - <NSThread: 0x717e7e0>{name = (null), num = 1}test - <NSThread: 0x8078a80>{name = (null), num = 3}线程停止

最后注意performSelector:onThread:withObject:waitUntilDone:方法中最后的waitUntilDone参数,如果传YES的话,当前线程会等待Selector在另一个线程中执行完毕后继续执行。

时间: 2024-10-19 11:21:09

iOS: NSObject中执行Selector的相关方法的相关文章

NSObject 中执行Selector 的相关方法

1. 对当前Run Loop中Selector Sources的取消 NSObject中的performSelector:withObject:afterDelay:方法将会在当前线程的Run Loop中根据afterDelay参数创建一个Timer,如果没有调用有inModes参数的方法,该Timer会运行在当前Run Loop的默认模式中,也就是NSDefaultRunLoopMode定义的模式中. performSelector:withObject:afterDelay:方法的使用看起来

iOS开发——淫技篇&amp;iOS开发中各种淫技总结(六)

iOS开发中各种淫技总结(六) swift中指针的使用 在 Swift 中,指针都使用一个特殊的类型来表示,那就是 UnsafePointer<T>.遵循了 Cocoa 的一贯不可变原则,UnsafePointer<T> 也是不可变的.当然对应地,它还有一个可变变体,UnsafeMutablePointer<T>.绝大部分时间里,C 中的指针都会被以这两种类型引入到 Swift 中:C 中 const 修饰的指针对应 UnsafePointer (最常见的应该就是 C

iOS多线程中,队列和执行的排列组合结果分析

本文是对以往学习的多线程中知识点的一个整理. 多线程中的队列有:串行队列,并发队列,全局队列,主队列. 执行的方法有:同步执行和异步执行.那么两两一组合会有哪些注意事项呢? 如果不是在董铂然博客园看到这边文章请 点击查看原文 提到多线程,也就是四种,pthread,NSthread,GCD,NSOperation 其中phtread是跨平台的.GCD和NSOperation都是常用的,后者是基于前者的. 但是两者区别:GCD的核心概念是将一个任务添加到队列,指定任务执行的方法,然后执行. NSO

Java中子类和父类相关方法的执行顺序

无意中看到下面一个题目,大家一起来看看最后的输出结果是什么.反正我看完之后,用IDE测试后感觉知识点得到巩固了. 1 /** 2 * 函数执行顺序测试 3 * Created by 萌小Q on 2017/5/17 0017. 4 */ 5 public class ExeSeqTest { 6 7 public static void main(String [] args){ 8 System.out.println(new B().getValue()); 9 } 10 static cl

iOS 8 动画执行过程中返回 Crash

之前项目里一直有个iOS8 [UIScrollView(UIScrollViewInternal) _notifyDidScroll] crash的问题,存在了很久,后来复现问题,返现是动画执行过程中执行pop 的问题 场景:在商品详情页加车之后  做了滑动到下面的商品推荐位置, 当动画还没有执行完毕, 就POP回上个页面,就会发生如下Crash 参考文档 https://stackoverflow.com/questions/26103756/uiscrollview-internal-con

iOS多线程中performSelector: 和dispatch_time的不同

iOS中timer相关的延时调用,常见的有NSObject中的performSelector:withObject:afterDelay:这个方法在调用的时候会设置当前runloop中timer,还有一种延时,直接使用NSTimer来配置任务. 这两种方式都一个共同的前提,就是当前线程里面需要有一个运行的runloop并且这个runloop里面有一个timer. 我们知道:只有主线程会在创建的时候默认自动运行一个runloop,并且有timer,普通的子线程是没有这些的.这样就带来一个问题了,有

iOS开发中的零碎知识点笔记 韩俊强的博客

每日更新关注:http://weibo.com/hanjunqiang  新浪微博 1.关联 objc_setAssociatedObject关联是指把两个对象相互关联起来,使得其中的一个对象作为另外一个对象的一部分. 2.tableView的beginUpdates 和 endUpdates 3.关于代码与storyBoard的自动布局 4.国际化与本地化,为了实现全球化 5.技巧 可以通过设置Scheme来设置app所运行的语言,你想要什么语言就是什么语言,而不用重新设置系统的语言. 6.i

ios开发中的4种数据持久化方式【二、数据库 SQLite3、Core Data 的运用】

               在上文,我们介绍了ios开发中的其中2种数据持久化方式:属性列表.归档解档.本节将继续介绍另外2种iOS持久化数据的方法:数据库 SQLite3.Core Data 的运用: 在本节,将通过对4个文本框内容的创建.修改,退出后台,再重新回到后台,来认识这两种持久化数据的方式.效果图如下[图1]: [图1 GUI界面效果图] [本次开发环境: Xcode:7.2     iOS Simulator:iphone6S plus   By:啊左]     一.数据库SQL

iOS开发中视图控制器ViewControllers之间的数据传递

iOS开发中视图控制器ViewControllers之间的数据传递 这里我们用一个demo来说明ios是如何在视图控制器之间传递重要的参数的.本文先从手写UI来讨论,在下一篇文章中讨论在storyboard中传递数据. 首先新建一个空工程,并添加一个根视图控制器类,如下图所示: # 在函数didFinishLunchingWithOption中添加几行代码,完成后如下: ? 1 2 3 4 5 6 7 8 9 10 11 12 - (BOOL)application:(UIApplication