首页 > 编程知识 正文

ios开发转型((开发过程经验转发)iOS PerformSelector)

时间:2023-05-03 18:00:33 阅读:123530 作者:3332

必须注意一点

[ selfperformselector 3360 @ selector (delay method ) with object :3358 www.Sina.com/after delay 33602.0 ]

名为property的参数在运行完成之前被强烈引用

基础用法

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

[ objperformselector : @ selector (play ) ] [ objperformselector : @ selector (play : ) withObject:@'ctdmt'] ' 在[ objperformselector : @ selector (play : with : ) with object : @ ' CT DMT ' with object 3360 @ ' kkd fbx ' ]编译阶段

Undeclared selector ' '执行的方法名称也是动态不确定的参数:

[ objperformselector : selector ]编译器只会警告您,由于不知道当前方法名称,因此可能会引起与内存泄漏相关的问题:

由于是performselectormaycausealeakbecauseitsselectorisunknown,因此在实际开发中,请使用performselector方法来避免运行时突然报告错误、找不到方法等问题

双延迟执行

[ objperformselector : @ selector (play ) with object : @ ' CT DMT ' after delay :4.f ]此方法延迟4秒钟后再运行play方法。 实际上,说到时间相关的处理,项目中经常使用的是NSTimer。 在Runloop中注册NSTimer后,Runloop会重复在适当的时间注册事件。 当然,Runloop并不是在准确的时间点触发事件以节约资源。

perform selector 3360 with object : after delay :在内部创建NSTimer,并将其添加到当前线程的Runloop中。 因此,在将该方法添加到子线程时,需要注意两个位置:

在子线程上执行时是否调用test方法

dispatch _ queue _ tqueue=dispatch _ get _ global _ queue (dispatch _ queue _ priority _ default,0 ); dispatch_async(queue,^ { [ selfperformselector : @ selector ] test ) with object : nilafterdelay 33602 ] ); 缺省情况下,子线程中的runloop没有启动,因此可以看到没有调用test方法。 使用run方法打开当前线程的runloop,但请注意run和延迟方法的执行顺序。

dispatch _ queue _ tqueue=dispatch _ get _ global _ queue (dispatch _ queue _ priority _ default,0 ); dispatch _ async (队列,^ { [ nsrunloopcurrentrunloop ] run }; [ selfperformselector : @ selector (test ) with object : nilafterdelay :2 ] ); 如果添加run方法,但不调用test方法,最后打印当前线程的runloop,则会找到:

timers=cf array0x 600002 a 8100 [0x 109 f67bb0] { type=mutable-small,count=1,values=(0: cfrunloooptimer0x 60000011 间隔=0,tolerance=0,nextfiredate=544280547 (1.98647892 @ 379501066754 )、callout=(delayedperform ) lzlearnining

交换run方法和performSelector延迟方法的顺序执行:

dispatch _ queue _ tqueue=dispatch _ get _ global _ queue (dispatch _ queue _ priority _ default,0 ); dispatch_async(queue,^{ [self perf

ormSelector:@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中也能执行。

三 异步执行

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

如何在不使用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];});

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

四 多参传递

一般在聊完这么多和performSelector相关的方法后,不要放松警惕,又一个怀疑人生的问题来了.

 

performSelector如何进行多值传输?

问题一听马上就能回答使用NSArray或者NSDictionary或者自定义Model的形式,但是我查到了一个很妙的方法:
因为在OC中调用一个方法实际上就是发送消息objc_msgSend:

{ NSNumber *age = [NSNumber numberWithInt:20]; NSString *name = @"ctdmt"; NSString *gender = @"女"; NSArray *friends = @[@"kkdfbx",@"亚呼呼"]; SEL selector = NSSelectorFromString(@"getAge:name:gender:friends:"); NSArray *array = @[age,name,gender,friends]; ((void(*)(id,SEL,NSNumber*,NSString*,NSString*,NSArray*)) objc_msgSend)(self,selector,age,name,gender,friends);}- (void)getAge:(NSNumber *)age name:(NSString *)name gender:(NSString *)gender friends:(NSArray *)friends{ NSLog(@"%d----%@---%@---%@",[age intValue],name,gender,friends[0]);}

导入#import <objc/message.h>即可。但是这种方式并不是oc封装的方法所以使用十分的不方便。
网上的第二种方法其实也是以NSArray的形式传值,然后创建NSInvocation的方式,将参数一一绑定。

-(id)performSelector:(SEL)aSelector withObject:(NSArray *)object{ //获得方法签名 NSMethodSignature *signature = [[self class] instanceMethodSignatureForSelector:aSelector]; if (signature == nil) { return nil; } //使用NSInvocation进行参数的封装 NSInvocation *invocation = [NSInvocation invocationWithMethodSignature:signature]; invocation.target = self; invocation.selector = aSelector; //减去 self _cmd NSInteger paramtersCount = signature.numberOfArguments - 2; paramtersCount = MIN(object.count, paramtersCount); for (int i = 0; i < paramtersCount; i++) { id obj = object[i]; if ([obj isKindOfClass:[NSNull class]]) continue; [invocation setArgument:&obj atIndex:i+2]; } [invocation invoke]; id returnValue = nil; if (signature.methodReturnLength > 0) { //如果有返回值的话,才需要去获得返回值 [invocation getReturnValue:&returnValue]; } return returnValue; } NSNumber *age = [NSNumber numberWithInt:20]; NSString *name = @"ctdmt"; NSString *gender = @"女"; NSArray *friends = @[@"kkdfbx",@"亚呼呼"]; SEL selector = NSSelectorFromString(@"getAge:name:gender:friends:"); NSArray *array = @[age,name,gender,friends]; [self performSelector:selector withObject:array];

NSInvocation我是在消息转发机制中认识的,所以这种方法类似于消息转发机制中的最后一层,多了创建NSInvocation对象的开销。而且本质上还是就NSArray进行转发。

版权声明:该文观点仅代表作者本人。处理文章:请发送邮件至 三1五14八八95#扣扣.com 举报,一经查实,本站将立刻删除。