關于NSInvocation的問題

這兩天接了個阿里的面試電話,有一個問題難到了,問了我關于NSInvocation的問題,當時真是不知道,后來趕緊補了下,發現NSInvocation主要還是用來解決多個參數如何傳遞的問題和直接拿到返回值,
以下是簡單的創建方法簽名的例子,可以看到既可以從類中獲取簽名,也可從實例中獲取。

SEL initSel = @selector(init);

  SEL allocSel = @selector(alloc);

  NSMethodSignature *initSig, *allocSig;

  //從實例中請求實例方法簽名

  initSig = [@"String" methodSignatureForSelector:initSel];

  //從類中請求實例方法簽名

  initSig = [NSString instanceMethodSignatureForSelector:initSel];

  //從類中請求實例方法簽名

  allocSig = [NSString methodSignatureForSelector:allocSel];

以下是網上摘抄的一部分代碼,試驗了下可以達到,但是不知道為什么會崩潰,沒找到原因

#import "ViewController.h"

@interface Target : NSObject

@end

@implementation Target

-(NSString *)saySomeThingA:(NSString *)a B:(NSString *)b C:(NSString *)c
{
    NSString *string = [NSString stringWithFormat:@"%@ and %@ and %@",a,b,c];
    return string;
}


-(id)performSelector:(SEL)aSelector withArguments:(NSArray *)arguments
{
    // 方法簽名中保存了方法的名稱/參數/返回值,協同NSInvocation來進行消息的轉發
    // 方法簽名一般是用來設置參數和獲取返回值的, 和方法的調用沒有太大的關系
    NSMethodSignature *signature = [[self class] instanceMethodSignatureForSelector:aSelector];
    NSInvocation *invocation = [NSInvocation invocationWithMethodSignature:signature];
    invocation.target = self;
    invocation.selector = aSelector;
    
    // invocation 有2個隱藏參數,所以 argument 從2開始
    if ([arguments isKindOfClass:[NSArray class]]) {
        //此處不能通過遍歷參數數組來設置參數,因為外界傳進來的參數個數是不可控的
        //因此通過numberOfArguments方法獲取的參數個數,是包含self和_cmd的,然后比較方法需要的參數和外界傳進來的參數個數,并且取它們之間的最小值
        NSInteger count = MIN(arguments.count, signature.numberOfArguments-2);
        for (int i = 0; i < count; i++) {
            const char *type = [signature getArgumentTypeAtIndex:2+i];
            // 需要做參數類型判斷然后解析成對應類型,這里默認所有參數均為OC對象
            if (strcmp(type, "@") == 0) {
                id argument = arguments[i];
                if ([argument isKindOfClass:[NSNull class]]) {
                    argument = nil;
                }
                [invocation setArgument:&argument atIndex:2+i];
            }
        }
    }
    [invocation invoke];
    
    //判斷當前調用的方法是否有返回值
    id renturnVal;
    if (strcmp(signature.methodReturnType, "@") == 0) {//有返回值
        //將返回值賦值給renturnVal
        [invocation getReturnValue:&renturnVal];
    }
    // 需要做返回類型判斷。比如返回值為常量需要包裝成對象,這里僅以最簡單的`@`為例
    return renturnVal;
}




@end


@implementation ViewController

- (void)viewDidLoad {
    [super viewDidLoad];
    
    //通過Block調用
    id returnVal = invokeBlock((id)^(NSString *a,NSString *b,NSString *c){
        return [NSString stringWithFormat:@"%@ and %@ and %@",a,b,c];
    },@[@"01",@"02",@"03"]);
    NSLog(@"%@",returnVal);
    
    //直接調用
    Target *targetnew = [Target new];
    NSString *str111 = [targetnew performSelector:@selector(saySomeThingA:B:C:) withArguments:@[@"曉明",@"在嗎",@"開門"]];
    NSLog(@"%@",str111);
    // Do any additional setup after loading the view, typically from a nib.
}


- (void)didReceiveMemoryWarning {
    [super didReceiveMemoryWarning];
    
    
    // Dispose of any resources that can be recreated.
}

static id invokeBlock(id block,NSArray *arguments){
    if (block == nil) {return nil;}
    id target = [block copy];
    const char *_Block_signature(void *);
    const char *signature = _Block_signature((__bridge void *)target);
    
    NSMethodSignature *methodSignature = [NSMethodSignature signatureWithObjCTypes:signature];
    NSInvocation *invocation = [NSInvocation invocationWithMethodSignature:methodSignature];
    invocation.target = target;
    
    // invocation 有1個隱藏參數,所以 argument 從1開始
    if ([arguments isKindOfClass:[NSArray class]]) {
        //此處不能通過遍歷參數數組來設置參數,因為外界傳進來的參數個數是不可控的
        //因此通過numberOfArguments方法獲取的參數個數,是包含self和_cmd的,然后比較方法需要的參數和外界傳進來的參數個數,并且取它們之間的最小值
        NSInteger count = MIN(arguments.count, methodSignature.numberOfArguments-1);
        for (int i = 0; i < count; i++) {
            const char *type = [methodSignature getArgumentTypeAtIndex:1+i];
            NSString *typeStr = [NSString stringWithUTF8String:type];
            // 需要做參數類型判斷然后解析成對應類型,這里默認所有參數均為OC對象
            if ([typeStr containsString:@"\""]) {
                type = [typeStr substringToIndex:1].UTF8String;
            }
            // 需要做參數類型判斷然后解析成對應類型,這里默認所有參數均為OC對象
            if (strcmp(type, "@") == 0) {
                id argument = arguments[i];
                [invocation setArgument:&argument atIndex:1 + i];
            }
        }
    }
    [invocation invoke];
    
    //判斷當前調用的方法是否有返回值
    id renturnVal;
    const char *type = methodSignature.methodReturnType;
    NSString *returnType = [NSString stringWithUTF8String:type];
    if ([returnType containsString:@"\""]) {
        type = [returnType substringToIndex:1].UTF8String;
    }
    if (strcmp(type, "@") == 0) {
        [invocation getReturnValue:&renturnVal];
    }
    // 需要做返回類型判斷。比如返回值為常量需要包裝成對象,這里僅以最簡單的`@`為例
    return renturnVal;
}


@end




最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 230,247評論 6 543
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 99,520評論 3 429
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 178,362評論 0 383
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,805評論 1 317
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 72,541評論 6 412
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 55,896評論 1 328
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,887評論 3 447
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 43,062評論 0 290
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 49,608評論 1 336
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 41,356評論 3 358
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 43,555評論 1 374
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 39,077評論 5 364
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,769評論 3 349
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 35,175評論 0 28
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 36,489評論 1 295
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 52,289評論 3 400
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 48,516評論 2 379

推薦閱讀更多精彩內容