基础用法

performSelecor响应了OC语言的动态性:延迟到运行时才绑定方法。当我们在使用以下方法时:

[obj performSelector:@selector(play)];
[obj performSelector:@selector(play:) withObject:@"李周"];
[obj performSelector:@selector(play:with:) withObject:@"李周" withObject:@"谢华华"];

编译阶段并不会去检查方法是否有效存在,只会给出警告:

Undeclared selector ''

如果要执行的方法名也是动态不确定的一个参数:

 [obj performSelector:selector];

编译器也只会提示说因为当前方法名未知可能会引起内存泄露相关问题:

PerformSelector may cause a leak because its selector is unknown

所以在实际开发中,为了避免运行时突然报错找不到方法等问题,少使用performSelector方法。

二 延迟执行

[obj performSelector:@selector(play) withObject:@"李周" afterDelay:4.f];

该方法将延迟4秒后再执行play方法。其实说到对时间方面的处理在项目中经常用到的是NSTimer:当一个NSTimer注册到Runloop后,Runloop会重复的在相应的时间点注册事件,当然Runloop为了节省资源并不会在准确的时间点触发事件。
而performSelector:withObject:afterDelay:其实就是在内部创建了一个NSTimer,然后会添加到当前线程的Runloop中。所以当该方法添加到子线程中时,需要格外的注意两个地方:

① 在子线程中执行会不会调用test方法

 dispatch_queue_t queue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0);dispatch_async(queue, ^{[self performSelector:@selector(test) withObject:nil afterDelay:2];
});

会发现test方法并没有被调用,因为子线程中的runloop默认是没有启动的状态。使用run方法开启当前线程的runloop,但是一定要注意run方法和执行该延迟方法的顺序。

 dispatch_queue_t queue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0);dispatch_async(queue, ^{[[NSRunLoop currentRunLoop] run];[self performSelector:@selector(test) withObject:nil afterDelay:2];
});

会发现即便添加了run方法,但是test方法还是没有被调用,在最后打印当前线程的runloop,会发现:

timers = <CFArray 0x6000002a8100 [0x109f67bb0]>{type = mutable-small, count = 1, values = (0 : <CFRunLoopTimer 0x6000001711c0 [0x109f67bb0]>{valid = Yes, firing = No, interval = 0, tolerance = 0, next fire date = 544280547 (1.98647892 @ 3795501066754), callout = (Delayed Perform) lZLearningFromInterviewController test (0x105ea0d9c / 0x104b2e2c0) (), context = <CFRunLoopTimer context 0x600000470080>}

子线程的runloop中确实添加了一个CFRunLoopTimer的事件,但是到最后都不会被执行。
将run方法和performSelector延迟方法调换顺序后运行:

 dispatch_queue_t queue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0);dispatch_async(queue, ^{[self performSelector:@selector(test) withObject:nil afterDelay:2];[[NSRunLoop currentRunLoop] run];
});

此时test方法会被调用,分别打印执行完performSelecor和run方法之后,发现在执行完performSelector方法后该timer事件会被添加到子线程的runloop中:

timers = <CFArray 0x6000000b3c80 [0x112956bb0]>{type = mutable-small, count = 1, values = (0 : <CFRunLoopTimer 0x60000016fc00 [0x112956bb0]>{valid = Yes, firing = No, interval = 0, tolerance = 0, next fire date = 544280800 (1.98171604 @ 4048676578329), callout = (Delayed Perform) lZLearningFromInterviewController test (0x10e88fd9c / 0x1

但是当执行完run方法之后,runloop中的timer事件已经是执行完的状态:

timers = <CFArray 0x6000000b3c80 [0x112956bb0]>{type = mutable-small, count = 0, values = ()},

所以在子线程中两者的顺序必须是先执行performSelector延迟方法之后再执行run方法。因为run方法只是尝试想要开启当前线程中的runloop,但是如果该线程中并没有任何事件(source、timer、observer)的话,并不会成功的开启。

② test方法中执行的线程

 [self performSelector:@selector(test) withObject:nil afterDelay:2];

如果在子线程中调用该performSelector延迟方法,会发现调用该延迟方法的子线程和test方法中执行的子线程是同一个,也就是说:

对于该performSelector延迟方法而言,如果在主线程中调用,那么test方法也是在主线程中执行;如果是在子线程中调用,那么test也会在该子线程中执行。

在回答完延迟方法之后,会将该方法和performSelector:withObject:作对比,那么performSelector:withObject:在不添加到子线程的Runloop中时是否能执行?
我当时想的是,performSelector:withObject:方法和延迟方法类似,只不过是马上执行而已,所以也需要添加到子线程的RunLoop中。

这么想是错的,performSelector:withObject:只是一个单纯的消息发送,和时间没有一点关系,不需要runloop帮忙。所以不需要添加到子线程的Runloop中也能执行;根本原因是afterDelay方式调用需要timer,timer调用需要runloop。

三 异步执行

有时候面试关于多线程的问题时,会提问说:

如何在不使用GCD和NSOperation的情况下,实现异步线程?

反正我第一反应就是:幸亏,把NSThread给我留下了!

所以能直接使用NSThread的三个方法:

NSThread *thread = [[NSThread alloc] initWithTarget:self selector:@selector(test) object:nil];
[NSThread detachNewThreadSelector:@selector(test) toTarget:self withObject:nil];
[NSThread detachNewThreadWithBlock:^{NSLog(@"block中的线程 ---- %@",[NSThread currentThread]);
}];

但是一般面试还会接着往下问:

如果也不使用NSThread已有的方法呢?

这个时候已经没有时间吐槽了只能接着想了...后来的后来我在perSelector的相关方法中找到了解答:

① performSelectorInBackground 后台执行

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

该方法一目了然,开启新的线程在后台执行test方法

②performSelector:onThread:在指定线程执行

[self performSelector:@selector(test) onThread:[NSThread currentThread] withObject:nil waitUntilDone:YES];

这个方法有一个thread参数是指定执行的线程,但是很奇怪当我使用自己创建的线程 [[NSThread alloc] init];时,并不会执行test方法,只有当使用[NSThread currentThread]时才会执行:

dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{[self performSelector:@selector(tests) onThread:[NSThread currentThread] withObject:nil waitUntilDone:NO];
});

还需要再考证考证这个方法的使用。

作者:李周
链接:https://www.jianshu.com/p/da010a008741
来源:简书
简书著作权归作者所有,任何形式的转载都请联系作者获得授权并注明出处。

1.参数包装成字典
2.自己实现对应方法 使用NSMethodSignure,NSInvoation

- (id)performSelector:(SEL)aSelectorwithObjects:(NSArray *)arguments{//1.根据SEL实例化方法签名NSMethodSignature *signature = [[self class]instanceMethodSignatureForSelector:aSelector];//2.判断方法是否存在if (signature == nil) {//抛出异常NSLog(@"不存在这个方法");return nil;}//3.通过类方法实例化NSInvaction对象,设置target,selectorNSInvocation *invocation = [NSInvocation invocationWithMethodSignature:signature];[invocation setTarget:self];[invocation setSelector:aSelector];//获取参数的个数,默认方法都有 self,_cmd两个参数NSInteger signatureParmCount = signature.numberOfArguments - 2;NSInteger resultCount = MIN(signatureParmCount, arguments.count);//设置方法参数for (NSInteger i = 0; i < resultCount; i++) {id argument = arguments[i];if ([argument isKindOfClass:[NSNull class]]) continue;[invocation setArgument:&argument atIndex:i+2];}[invocation invoke];//返回值,获取返回值的长度,大于0表示有返回值id returnArgument = nil;if (signature.methodReturnLength) {[invocation getReturnValue:&returnArgument];}return returnArgument;
}

3.objc_msgsend

    NSString *str = @"字符串objc_msgSend";NSNumber *num = @20;NSArray *arr = @[@"数组值1", @"数组值2"];SEL sel = NSSelectorFromString(@"ObjcMsgSendWithString:withNum:withArray:");((void (*) (id, SEL, NSString *, NSNumber *, NSArray *)) objc_msgSend) (self, sel, str, num, arr);

防止按钮多次点击

这种方式是在0.2秒内取消之前的点击事件,以做到防止多次点击。

-(void)completeClicked:(UIButton *)sender{[[self class] cancelPreviousPerformRequestsWithTarget:self selector:@selector(buttonClick:) object:sender];[self performSelector:@selector(buttonClick:) withObject:sender afterDelay:0.2f];
}

这种方式是在点击后设为不可被点击的状态,1秒后恢复

-(void)buttonClicked:(id)sender{self.button.enabled =NO;[selfperformSelector:@selector(changeButtonStatus)withObject:nilafterDelay:1.0f];//防止重复点击}-(void)changeButtonStatus{self.button.enabled =YES;
}

performSelector 的使用相关推荐

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

    iOS中timer相关的延时调用,常见的有NSObject中的performSelector:withObject:afterDelay:这个方法在调用的时候会设置当前runloop中timer,还有 ...

  2. 关于 performSelector 的一些小探讨

    本文首发在我的个人博客: blog.shenyuanluo.com,喜欢的朋友欢迎订阅. 考虑以下代码,最终会输出什么? 例子①: - (void)viewDidLoad {[super viewDi ...

  3. performSelector may cause a leak because its selector is unknown

    转自:http://www.jianshu.com/p/6517ab655be7 问题 我在 ARC 模式下编译出了这个 warning: "performSelector may caus ...

  4. [Xcode 实际操作]八、网络与多线程-(19)使用RunLoop使PerformSelector方法延迟动作的执行...

    目录:[Swift]Xcode实际操作 本文将演示使用RunLoop使PerformSelector方法延迟动作的执行. 在项目导航区,打开视图控制器的代码文件[ViewController.swif ...

  5. 初探swift语言的学习笔记十一(performSelector)

    作者:fengsh998 原文地址:http://blog.csdn.net/fengsh998/article/details/35842441 转载请注明出处 如果觉得文章对你有所帮助,请通过留言 ...

  6. Objective-C中一种消息处理方法performSelector: withObject:

    Objective-C中调用函数的方法是"消息传递",这个和普通的函数调用的区别是,你可以随时对一个对象传递任何消息,而不需要在编译的时候声明这些方法.所以Objective-C可 ...

  7. NSObject中的performSelector用法简介

    - (id)performSelector:(SEL)aSelector Description [说明] Sends a specified message to the receiver and ...

  8. iOS 网络与多线程--7.Performselector消息处理方法

    创建一个IOSApp类 IOSApp.h文件 1 #import <Foundation/Foundation.h> 2 3 @interface IOSApp : NSObject 4 ...

  9. performSelector:withObject:afterDelay: 精要概览(持续更新)

    - (void)performSelector:(SEL)aSelector withObject:(id)anArgument afterDelay:(NSTimeInterval)delay; - ...

  10. 如何在延迟后触发一个块,比如-performSelector:withObject:afterDelay:?

    有没有办法在延迟后调用带有原始参数的块,比如使用performSelector:withObject:afterDelay:但是使用像int / double / float这样的参数? #1楼 Bl ...

最新文章

  1. 查看oracle的表空间使用,查看Oracle的表空间的使用情况
  2. 深度学习100问:图像语义分割有哪些经典的上采样方法?
  3. c++中的变量作用范围
  4. 科大星云诗社动态20210830
  5. 【2020蓝桥杯】Python组真题解析 - 第十一届蓝桥杯
  6. git如何切换分支_如何让 Jenkins 支持选择 git 分支进行构建
  7. 一种新的8B/10B编解码硬件设计方法
  8. linux程序无法获取cpu资源,Linux系统编程获取系统的CPU资源
  9. php ajax ip,php / ajax REMOTE_ADDR设置为伪网络适配器的IP
  10. SQL语法--DML
  11. 利用正则表达式 替换字符串中多个 URL
  12. 数据产品经理面试提问集
  13. 读书笔记-互联网鲇鱼法则
  14. 西门子转以太网模块连接S7-300以太网驱动
  15. echarts 弹出放大_vue中使用v-chart时放大缩小屏幕,echarts图自适应
  16. mysql实现分组查询每个班级的前三名、每门课程的前三名
  17. 用纯CSS实现八卦太极图
  18. 小蓝本 第一本《因式分解技巧》 第一章 提公因式 笔记(第一天)
  19. 机器学习中的Encoder和Decoder到底是什么
  20. 17 Equations That Changed the World

热门文章

  1. 虚拟机通过jumpserver登录服务器,搭建 JumpServer 堡垒机管理数万台游戏服务器
  2. C#操作GDAL——(1)空间参考相关
  3. redis集群数据迁移方式、宕机
  4. 如何制定医院病区6S管理考核标准?
  5. java 异步_浅谈Java异步编程
  6. kong笔记——自定义插件开发
  7. 物联网开发之智慧农业解决方案
  8. 万事俱备,只欠价格? — 华为 MateBook 13 长测
  9. 北大计算机系录取通知书,五年级写出小游戏,广附高一学生拿到北大预录取通知书...
  10. php fitnesse,自动化测试框架FitNesse -- 搭建CsLim