背景:直接 對(duì)
NSArray
/NSMutableArray
的方法交換時(shí),總是不成功
原因
NSArray
/NSMutableArray
是抽象類构诚,使用類簇創(chuàng)建颠放,真正的實(shí)現(xiàn)類是__NSArrayI
/__NSArrayM
。
例子
為了避免數(shù)組越界轴咱,使用Runtime
交換NSArray
的objectAtIndex:
方法到我們定義好判斷的swiObjectAtIndex:
方法铭乾。
意義:數(shù)組越界的判斷邏輯集中到一個(gè)地方剪廉,減少代碼量。
- 這是
NSArray
分類的邏輯炕檩,主要功能也只有一個(gè)斗蒋,objectAtIndex:
和swiObjectAtIndex:
交換。
+ (void)load
{
[[self class] swizzlingMethodWithCurrentClass:NSClassFromString(@"__NSArrayI")
OrigSelector:@selector(objectAtIndex:)
SwizzleSelector:@selector(swiObjectAtIndex:)];
}
- (id)swiObjectAtIndex:(NSInteger)index
{
if (index >= 0 && index < self.count)
{
return [self swiObjectAtIndex:index];
}
NSLog(@"index is outof array!!");
return nil;
}
+ (void)swizzlingMethodWithCurrentClass:(Class)currentClass
OrigSelector:(SEL)origSelector
SwizzleSelector:(SEL)swizzleSelector
{
Method originalMethod = class_getInstanceMethod(currentClass, origSelector);
Method swizzleMethod = class_getInstanceMethod(currentClass, swizzleSelector);
// 首先檢查原方法是否存在
BOOL didAddMethod = class_addMethod(currentClass, origSelector, method_getImplementation(swizzleMethod), method_getTypeEncoding(swizzleMethod));
if (didAddMethod) //原方法不存在捧书,需要先添加方法吹泡,再替換骤星。class_replaceMethod 等價(jià)于class_addMethod和method_setImplementation
{
class_replaceMethod(currentClass, swizzleSelector, method_getImplementation(originalMethod), method_getTypeEncoding(originalMethod));
}
else // 原方法存在经瓷,直接替換
{
method_exchangeImplementations(originalMethod, swizzleMethod);
}
}
最終使用的時(shí)候,就不會(huì)崩潰了洞难。
NSArray *array = @[@"1",@"2"];
NSString *str = [array objectAtIndex:4];
NSLog(@"str --- %@",str);
注意:
1舆吮、方法交換的時(shí)候,使用的是NSClassFromString(@"__NSArrayI")
而不是NSArray
队贱,其實(shí)最終執(zhí)行方法的是__NSArrayI
色冀。NSMutableArray
對(duì)應(yīng)__NSArrayM
。
2柱嫌、如何得知__NSArrayI
/__NSArrayM
類型锋恬。對(duì)!讓它崩潰一次编丘,lldb
中reason
會(huì)打印出來(lái)与学。
參考:
http://www.reibang.com/p/f58a9e4be184
解釋得很細(xì)致的交換方法
交換方法的各種類型