转载iOS DeveloperiOS Swift && Objective-C

NSInvocation的学习及封装(赠送异常捕获)

2017-01-18  本文已影响169人  ZhengYaWei

在做网页和原生相互调用的时候,尤其是在js调用原生进行参数传递的时候,通常会通过改变h5中的location.href,进而在网页加载的时候,根据location.href来区分究竟是何种操作,应该调用什么方法,传递什么参数。这是通常会大量的使用到performSelector: 这个方法,但是这个方法有个局限性,最多只能传递两个参数,当然也可以通过数组保存多个参数。另外还可以通过NSInvocation来实现。接下来就想来学习NSInvocation的基本知识。然后再说明如何封装一个NSInvocation类别方法。创建invocation一般有四个步骤:
1.签名 包含信息:方法名称,返回值,参数,谁谁有这个方法等信息,签名和调用方法没有关系
2.创建NSInvocation对象
3.参数等相关信息
4.执行invocation方法
这里要说明一个注意点,我们平时使用的方法,默认都会有两个参数(self和_cmd)传递进去,所以在传递参数的时候,要从2开始。

- (void)invocation2{
    NSMethodSignature *signature = [ViewController instanceMethodSignatureForSelector:@selector(callWithNumber:andPerson:withPerson2:)];
    NSInvocation *invocation  = [NSInvocation invocationWithMethodSignature:signature];
    invocation.target = self;
    invocation.selector = @selector(callWithNumber:andPerson:withPerson2:);
    NSString *number = @"152222";
    NSString *p1 = @"jack";
    NSString *p2 = @"jony";
    //index表示第几个参数,注意0和1已经被占用了(self和_cmd)
    //我们传递参数的时候要从2开始
    [invocation setArgument:&number atIndex:2];
    [invocation setArgument:&p1 atIndex:3];
    [invocation setArgument:&p2 atIndex:4];
    [invocation invoke];
}

被调用的方法。

- (void)callWithNumber:(NSString *)number andPerson:(NSString *)person  withPerson2:(NSString *)person2{
    NSLog(@"----%@  --------%@   -----%@",number,person,person2);
}

接下来看看如何封装一个自己的类别,可以实现我们的需求。可以参照系统的performactSelector方法设计方法接口,但是为了能传递无限多的参数,我们传递数组,直接将参数保存在数组中。首先创建一个NSObject的分类,然后对外提供一个这样的方法:

- (id)zw_performSelector:(SEL)aSelector withObjects:(NSArray *)objects;

方法实现代码。

- (id)zw_performSelector:(SEL)aSelector withObjects:(NSArray *)objects {
    // 通过选择器获取方法签名
    //0.签名 包含信息:方法名称,返回值,参数,谁谁有这个方法等信息,签名和调用方法没有关系
    NSMethodSignature *signature = [[self class] instanceMethodSignatureForSelector:aSelector];
    if (signature == nil) {
        NSString *reason = [NSString stringWithFormat:@"***** The method[%@] is not find ******", NSStringFromSelector(aSelector)];
        @throw [NSException exceptionWithName:@"错误" reason:reason userInfo:nil];
    }
    NSInvocation *invocation = [NSInvocation invocationWithMethodSignature:signature];
    invocation.target = self;
    invocation.selector = aSelector;
    //index表示第几个参数,注意0和1已经被占用了(self和_cmd)
    //我们传递参数的时候要从2开始
    NSInteger paras = signature.numberOfArguments - 2;
    paras = MIN(paras, objects.count);
    for (NSInteger i = 0; i < paras; i++) {
        id object = objects[i];
        if ([object isKindOfClass:[NSNull class]]) {
            continue;
        }
        [invocation setArgument:&object atIndex:i+2];
    }
    // 调用方法
    [invocation invoke];
    
    id returnValue = nil;
    if (signature.methodReturnLength) {
        //获取返回值
        [invocation getReturnValue:&returnValue];
    }
    return returnValue;
}

外部调用形式。外部调用的时候遇到了一个问题,但是不是很清楚,问题在代码中有说明。希望知道的人帮忙解答下。

- (void)viewDidLoad {
    [super viewDidLoad];
    SEL sel = NSSelectorFromString(@"testWithStr:");
    NSString *result = [self zw_performSelector:sel withObjects:@[@"ssss"]];
    NSLog(@"%@",result);
    
}
- (NSString *)testWithStr:(NSString *)str{
    NSLog(@"%@",str);
    NSString *tempStr = [NSString stringWithFormat:@"%@===",str];
    return tempStr;
    //如果这样写返回值的话,程序会崩溃,不知道原因,希望知道的可以帮忙解释一下。
    return [NSString stringWithFormat:@"%@====",str];
}

由于分别方法的封装用到了捕获异常,这里就简单提一下。很多时候程序崩溃了,系统不会打印相关错误信息。我们可以利用捕获异常记录崩溃相关的信息。在用户使用程序的时候,如果出现错误,可以将捕获的异常相关信息保存到本地,到用户下次再登录的时候,将捕获的相关错误信息上传到服务器。一般对捕获异常的处理都是放在AppDelegate中的,简单的看看在实际开发过程中中是怎样运用的。下面有相关注释。

- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {
    
    NSSetUncaughtExceptionHandler(&handler);
    
    return YES;
}
void handler(NSException *exception){
    //一般的处理是这样的:拼接异常信息,然后保存到沙盒中,下次进入程序中检查沙盒中是否有异常,有的话直接上传给服务器
    NSArray *arr = [exception callStackSymbols];//获取当前调用栈信息
    NSString *reason = [exception reason];//崩溃原因 重要
    NSString *name = [exception name];//异常类型
    NSLog(@"%@  %@  %@",arr,reason,name);
}
上一篇 下一篇

猜你喜欢

热点阅读