热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

iOSperformSelector多参数传递解决方案以及objc_msgSend的使用注意事项

1.iOSperformSelector多参数传递解决方案以及objc_msgSend的使用注意事项 https:blog.csdn.netglt_codearticledetai

1.iOS performSelector多参数传递解决方案以及objc_msgSend的使用注意事项

 

https://blog.csdn.net/glt_code/article/details/77584683

 

iOS performSelector多参数传递解决方案


以及objc_msgSend的使用注意事项

 


iOS中使用performSelector:withObject:withObject:方法最多传递两个参数

[self performSelectorOnMainThread:<#(nonnull SEL)#> withObject:<#(nullable id)#> waitUntilDone:<#(BOOL)#>]

 


解决方案:1. 使用NSInvocation进行消息转发从而实现对performSelector的多参数传递

2. 使用runtime中的objc_msgSend进行消息的发送

 

方案一:

以下是对NSObject类的扩展方法:

NS_REQUIRES_NIL_TERMINATION:是对多参数传递值得一个宏

va_list args:定义一个指向个数可变的参数列表指针;

va_start(args,object):object为第一个参数,也就是最右边的已知参数,这里就是获取第一个可选参数的地址.使参数列表指针指向函数参数列表中的第一个可选参数,函数参数列表中参数在内存中的顺序与函数声明时的顺序是一致的。

va_arg(args,id):返回参数列表中指针所指的参数,返回类型为id,并使参数指针指向参数列表中下一个参数。

va_end(args):清空参数列表,并置参数指针args无效。

其他说明见代码注释 Object分类

-(id)glt_performSelector:(SEL)selector withObject:(id)object,...NS_REQUIRES_NIL_TERMINATION;
{
//根据类名以及SEL 获取方法签名的实例
NSMethodSignature *signature = [[self class] instanceMethodSignatureForSelector:selector];
if (signature == nil) {
NSLog(@"--- 使用实例方法调用 为nil ---");
signature = [self methodSignatureForSelector:selector];
if (signature == nil) {
NSLog(@"使用类方法调用 也为nil, 此时return");
return nil;
}
}
//NSInvocation是一个消息调用类,它包含了所有OC消息的成分:target、selector、参数以及返回值。
NSInvocation *invocation = [NSInvocation invocationWithMethodSignature:signature];
invocation.target = self;
invocation.selector = selector;
NSUInteger argCount = signature.numberOfArguments;
// 参数必须从第2个索引开始,因为前两个已经被target和selector使用
argCount = argCount > 2 ? argCount - 2 : 0;
NSMutableArray *objs = [NSMutableArray arrayWithCapacity:0];
if (object) {
[objs addObject:object];
va_list args;
va_start(args, object);
while ((object = va_arg(args, id))){
[objs addObject:object];
}
va_end(args);
}
if (objs.count != argCount){
NSLog(@"--- objs.count != argCount! please check it! ---");
return nil;
}
//设置参数列表
for (NSInteger i = 0; i id obj = objs[i];
if ([obj isKindOfClass:[NSNull class]]) {
continue;
}
[invocation setArgument:&obj atIndex:i+2];
}
[invocation invoke];
//获取返回值
id returnValue = nil;
if (signature.methodReturnLength != 0 && signature.methodReturnLength) {
[invocation getReturnValue:&signature];
}
return returnValue;
}


 


shareExtension调用

 UIResponder* respOnder= self;

    NSString *urlStr = [NSString stringWithFormat:@"scheme://host/path%@",url];

   while ((respOnder= [responder nextResponder]) != nil) {

       if ([responder respondsToSelector:@selector(openURL:)] == YES && [responder isKindOfClass:[UIApplication class]]) {

          [responder performSelector:@selector(openURL:) withObject:[NSURL URLWithString:urlStr]];

      }else if([responder isKindOfClass:[UIApplication class]] && [responder respondsToSelector:@selector(openURL: options: completionHandler:)]) {

//           [responder performSelector:@selector(openURL: options: completionHandler:) withObject:[NSURL URLWithString:urlStr] withObject:@{}];

          void (^block)(void) = ^{

                  

              };

          [responder glt_performSelector:@selector(openURL: options: completionHandler:) withObject:[NSURL URLWithString:urlStr], @{},block,nil];

       }

    }

    [self.extensionContext completeRequestReturningItems:@[] completionHandler:nil];


注意:在Swift中没有NSMethodSignature和NSInvocation方法,暂时只能通过桥接解决,Swift中给出了这样一句:
NS_SWIFT_UNAVAILABLE("NSInvocation and related APIs not available")
如果在Swift中谁有好的方法可以扩展,请求大神告知!不胜感激!

方案二:

objc_msgSend :

objc_msgSend(<#id self#>, <#SEL op, ...#>)
概念:其实每个类中都有一张方法列表去存储这个类中有的方法,当发出objc_msgSend方法时候,就会顺着列表去找这个方法是否存在,如果不存在,则向该类的父类继续查找,直到找到位置。如果始终没有找到方法,那么就会进入到消息转发机制;objc_msgSend被分为2个过程:1)在cache中寻找SEL。2)在MethodTable寻找SEL。

具体使用介绍:

在一个列中有以下两个方法:

@implementation CustomClass

-(void)fun
{
NSLog(@"fun");
}

-(void)eat:(NSString *)food say:(NSString *)some
{
NSLog(@"%@ %@",food, some);
}

@end

当我们使用的时候,使用objc_msgSend进行调用,如下:这样便调用了类中的fun方法

TestClass *cls = [[TestClass alloc] init];

objc_msgSend(cls, @selector(fun)); //错误写法(arm64崩溃偶尔发生)

((void (*)(id, SEL))objc_msgSend)(cls, @selector(fun)); //正确写法

//具体原因见下面解释 objc_msgSend arm64 崩溃问题


使用注意1:使用objc_msgSend crash解决方案

 

 

使用注意2:objc_msgSend arm64 崩溃问题
之前一直用objc_msgSend,但是没注意apple的文档提示,所以突然objc_msgSend crash了。
按照文档 64-Bit Transition Guide for Cocoa Touch 给出了以下代码片段:

- (int) doSomething:(int) x { ... }
- (void) doSomethingElse {
int (*action)(id, SEL, int) = (int (*)(id, SEL, int)) objc_msgSend;
action(self, @selector(doSomething:), 0);
}
所以必须先定义原型才可以使用,这样才不会发生崩溃,调用的时候则如下:
void (*glt_msgsend)(id, SEL, NSString *, NSString *) = (void (*)(id, SEL, NSString *, NSString *))objc_msgSend;

glt_msgsend(cls, @selector(eat:say:), @"123", @"456");
————————————————
版权声明:本文为CSDN博主「高刘通」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。
原文链接:https://blog.csdn.net/glt_code/article/details/77584683

 



推荐阅读
  • 向QTextEdit拖放文件的方法及实现步骤
    本文介绍了在使用QTextEdit时如何实现拖放文件的功能,包括相关的方法和实现步骤。通过重写dragEnterEvent和dropEvent函数,并结合QMimeData和QUrl等类,可以轻松实现向QTextEdit拖放文件的功能。详细的代码实现和说明可以参考本文提供的示例代码。 ... [详细]
  • 使用nodejs爬取b站番剧数据,计算最佳追番推荐
    本文介绍了如何使用nodejs爬取b站番剧数据,并通过计算得出最佳追番推荐。通过调用相关接口获取番剧数据和评分数据,以及使用相应的算法进行计算。该方法可以帮助用户找到适合自己的番剧进行观看。 ... [详细]
  • 如何自行分析定位SAP BSP错误
    The“BSPtag”Imentionedintheblogtitlemeansforexamplethetagchtmlb:configCelleratorbelowwhichi ... [详细]
  • Spring源码解密之默认标签的解析方式分析
    本文分析了Spring源码解密中默认标签的解析方式。通过对命名空间的判断,区分默认命名空间和自定义命名空间,并采用不同的解析方式。其中,bean标签的解析最为复杂和重要。 ... [详细]
  • 本文分享了一个关于在C#中使用异步代码的问题,作者在控制台中运行时代码正常工作,但在Windows窗体中却无法正常工作。作者尝试搜索局域网上的主机,但在窗体中计数器没有减少。文章提供了相关的代码和解决思路。 ... [详细]
  • 目录实现效果:实现环境实现方法一:基本思路主要代码JavaScript代码总结方法二主要代码总结方法三基本思路主要代码JavaScriptHTML总结实 ... [详细]
  • Java容器中的compareto方法排序原理解析
    本文从源码解析Java容器中的compareto方法的排序原理,讲解了在使用数组存储数据时的限制以及存储效率的问题。同时提到了Redis的五大数据结构和list、set等知识点,回忆了作者大学时代的Java学习经历。文章以作者做的思维导图作为目录,展示了整个讲解过程。 ... [详细]
  • 本文介绍了OC学习笔记中的@property和@synthesize,包括属性的定义和合成的使用方法。通过示例代码详细讲解了@property和@synthesize的作用和用法。 ... [详细]
  • 本文讨论了一个关于cuowu类的问题,作者在使用cuowu类时遇到了错误提示和使用AdjustmentListener的问题。文章提供了16个解决方案,并给出了两个可能导致错误的原因。 ... [详细]
  • Android JSON基础,音视频开发进阶指南目录
    Array里面的对象数据是有序的,json字符串最外层是方括号的,方括号:[]解析jsonArray代码try{json字符串最外层是 ... [详细]
  • 本文详细介绍了Spring的JdbcTemplate的使用方法,包括执行存储过程、存储函数的call()方法,执行任何SQL语句的execute()方法,单个更新和批量更新的update()和batchUpdate()方法,以及单查和列表查询的query()和queryForXXX()方法。提供了经过测试的API供使用。 ... [详细]
  • 本文详细介绍了如何使用MySQL来显示SQL语句的执行时间,并通过MySQL Query Profiler获取CPU和内存使用量以及系统锁和表锁的时间。同时介绍了效能分析的三种方法:瓶颈分析、工作负载分析和基于比率的分析。 ... [详细]
  • 本文介绍了如何使用Express App提供静态文件,同时提到了一些不需要使用的文件,如package.json和/.ssh/known_hosts,并解释了为什么app.get('*')无法捕获所有请求以及为什么app.use(express.static(__dirname))可能会提供不需要的文件。 ... [详细]
  • Spring常用注解(绝对经典),全靠这份Java知识点PDF大全
    本文介绍了Spring常用注解和注入bean的注解,包括@Bean、@Autowired、@Inject等,同时提供了一个Java知识点PDF大全的资源链接。其中详细介绍了ColorFactoryBean的使用,以及@Autowired和@Inject的区别和用法。此外,还提到了@Required属性的配置和使用。 ... [详细]
  • IOS开发之短信发送与拨打电话的方法详解
    本文详细介绍了在IOS开发中实现短信发送和拨打电话的两种方式,一种是使用系统底层发送,虽然无法自定义短信内容和返回原应用,但是简单方便;另一种是使用第三方框架发送,需要导入MessageUI头文件,并遵守MFMessageComposeViewControllerDelegate协议,可以实现自定义短信内容和返回原应用的功能。 ... [详细]
author-avatar
Lanboream
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有