2022-03-15 iOS OC常见崩溃和防止崩溃方案

崩溃方案:

JJException
AvoidCrash

用到3个知识点

1、消息转发机制


@implementation HelloClass
//这里没啥用
-(BOOL)respondsToSelector:(SEL)aSelector{
    bool a= [super respondsToSelector:aSelector];
    return a;
}
//如果方法没有实现,默认返回false
//如果返回false,就会走消息转发
+(BOOL)resolveInstanceMethod:(SEL)sel{
    bool a = [super resolveInstanceMethod:sel];
    return a;
}
//默认返回空
//又被称为快速消息转发。
// 如果为空,走慢速消息转发,继续转发消息
-(id)forwardingTargetForSelector:(SEL)aSelector{
    id a = [super forwardingTargetForSelector:aSelector];
    return a;
}
//默认实现是崩溃
//并且不能用try-catch捕获
-(void)forwardInvocation:(NSInvocation *)anInvocation{
    [super forwardInvocation:anInvocation];
    NSLog(@"");
}
// 默认一般普通方法是返回空的。
// 如果是协议方法,没有实现,不会反回空。
//反回空,到这里就会崩溃了
//如果这里返回了签名,会再次调用resolveInstanceMethod:(SEL)sel判断是否实现
//如果仍然没有实现,就会走到fowardInvocation:
-(NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector{
    NSMethodSignature *a =[super methodSignatureForSelector:aSelector];
    return a;
}
@end

2、关联对象内存管理

通过对象dealloc时,关联对象也会dealloc的特性,可以给dealloc插入一些实现。


static const char DeallocNSObjectKey;

/**
 Observer the target middle object
 */
@interface DeallocStub : NSObject

@property (nonatomic,readwrite,copy) void(^deallocBlock)(void);

@end

@implementation DeallocStub

- (void)dealloc {
    if (self.deallocBlock) {
        self.deallocBlock();
    }
    self.deallocBlock = nil;
}

@end

@implementation NSObject (DeallocBlock)



- (void)isd_deallocBlock:(void(^)(void))block{
    @synchronized(self){
         // 用数组,  保证每一个block都能执行
        
        NSMutableArray* blockArray = objc_getAssociatedObject(self, &DeallocNSObjectKey);
        if (!blockArray) {
            blockArray = [NSMutableArray array];
             objc_setAssociatedObject(self, &DeallocNSObjectKey, blockArray, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
        }
        
        
        
        DeallocStub *stub = [DeallocStub new];
        stub.deallocBlock = block;
        [blockArray addObject:stub];
    }
}

@end

3、HOOK方法交换



void isd_swizzleClassMethod(Class cls, SEL originSelector, SEL swizzleSelector)
{
    if (!cls) {
        return;
    }
    Method originalMethod = class_getClassMethod(cls, originSelector);
    Method swizzledMethod = class_getClassMethod(cls, swizzleSelector);

    
    Class metacls = objc_getMetaClass(NSStringFromClass(cls).UTF8String);
    if (class_addMethod(metacls,
                        originSelector,
                        method_getImplementation(swizzledMethod),
                        method_getTypeEncoding(swizzledMethod)) ) {
        /* swizzing super class method, added if not exist */
        class_replaceMethod(metacls,
                            swizzleSelector,
                            method_getImplementation(originalMethod),
                            method_getTypeEncoding(originalMethod));
        
    } else {
        /* swizzleMethod maybe belong to super */
        class_replaceMethod(metacls,
                            swizzleSelector,
                            class_replaceMethod(metacls,
                                                originSelector,
                                                method_getImplementation(swizzledMethod),
                                                method_getTypeEncoding(swizzledMethod)),
                            method_getTypeEncoding(originalMethod));
    }
}



void isd_swizzleInstanceMethod(Class cls, SEL originSelector, SEL swizzleSelector)
{
    if (!cls) {
        return;
    }

 
    /* if current class not exist selector, then get super*/
    Method originalMethod = class_getInstanceMethod(cls, originSelector);
    Method swizzledMethod = class_getInstanceMethod(cls, swizzleSelector);

    /* add selector if not exist, implement append with method */
    //调用originSelector-》swizzledMethod
    if (class_addMethod(cls,
                        originSelector,
                        method_getImplementation(swizzledMethod),
                        method_getTypeEncoding(swizzledMethod)) ) {
        /* replace class instance method, added if selector not exist */
        /* for class cluster , it always add new selector here */
        //调用swizzleSelector-》originalMethod

        class_replaceMethod(cls,
                            swizzleSelector,
                            method_getImplementation(originalMethod),
                            method_getTypeEncoding(originalMethod));

    } else {
        /* swizzleMethod maybe belong to super */
        //swizzleSelector-》originalMethod
        //originSelector-》swizzledMethod
        class_replaceMethod(cls,
                            swizzleSelector,
                            class_replaceMethod(cls,
                                                originSelector,
                                                method_getImplementation(swizzledMethod),
                                                method_getTypeEncoding(swizzledMethod)),
                            method_getTypeEncoding(originalMethod));
    }
}

崩溃和保护方案

1、nstimer。aTarget 类没有实现对应的方法

hook NSTimer

+ (NSTimer *)scheduledTimerWithTimeInterval:(NSTimeInterval)ti target:(id)aTarget selector:(SEL)aSelector userInfo:(nullable id)userInfo repeats:(BOOL)yesOrNo;

方法。

2, nsnotfication奔溃, 没有移除观察者
观察者dealloc释放时候,没有从消息中心移除自己

hook NSNotificationCenter

 - (void)addObserver:(id)observer selector:(SEL)aSelector name:(nullable NSNotificationName)aName object:(nullable id)anObject;

方法。

给observer 对象添加关联对象。

observer对象释放-》 关联对象在dealloc-〉绑定block,关联对象dealloc时调用block,消息中心移除观察者。

3, kvo崩溃
多移除: it is not registered as an observer.、
少移除都会崩溃:deallocated while key value observers were still registered with it

hook

addObserver:forKeyPath:options:context:
removeObserver:forKeyPath:
removeObserver:forKeyPath:context:

方法。
通过关联对象。 记录 已经添加的 observer 和keypath。

4, 找不到方法崩溃

hook NSObject

methodSignatureForSelector:
forwardInvocation:

方法。



@implementation NSObject (UnrecognizedSelectorHook)

+ (void)xxx_swizzleUnrecognizedSelector{
    xxx_swizzleInstanceMethod(self, @selector(methodSignatureForSelector:), @selector(methodSignatureForSelectorSwizzled:));
    xxx_swizzleInstanceMethod(self, @selector(forwardInvocation:), @selector(forwardInvocationSwizzled:));
}

- (NSMethodSignature*)methodSignatureForSelectorSwizzled:(SEL)aSelector {
    NSMethodSignature* methodSignature = [self methodSignatureForSelectorSwizzled:aSelector];
    // 协议中的方法,但是没有实现, 默认返回非空
    if (methodSignature) {
        return methodSignature;
    }
    
    // 其他情况, 默认返回空
    
    //原始的默认实现NSObject 函数指针
    IMP originIMP = class_getMethodImplementation([NSObject class], @selector(methodSignatureForSelector:));
    
    // 当前类自己的实现
    IMP currentClassIMP = class_getMethodImplementation(self.class, @selector(methodSignatureForSelector:));
    
    // If current class override methodSignatureForSelector return nil
    //当前类自己重新实现了methodSignatureForSelector:方法;
    if (originIMP != currentClassIMP){
        return nil;
    }
    
    // Customer method signature
    // void xxx(id,sel,id)
    //随便返回点啥签名, 只要forwardInvocation被hook拦截住不调用就不会崩溃
    return [NSMethodSignature signatureWithObjCTypes:"v@:@"];
}



- (void)forwardInvocationSwizzled:(NSInvocation*)invocation{
    //forwardInvocation:
    NSString* message = [NSString stringWithFormat:@"Unrecognized selector class:%@ and selector:%@",NSStringFromClass(self.class),NSStringFromSelector(invocation.selector)];
    handleCrashException(xxxCrashProtectionUnrecognizedSelector,message,@{});
}


@end

还有其他方案:

HOOK
NSNULL、 NSObject、NSNumber、NSString
NSNULL类的 forwardingTargetForSelector:方法

比如对NSNumber调用NSString的方法,可以forwardingTargetForSelector:返回NSString对象。
NSString也一样, 原对象转化NSNumber对象返回。
NSNull @"", @[], @{}, @0 respondsToSelector判断,字符串、数组、字典、number哪个实现了原来selector,就返回对应固定对象,保证不崩溃。
NSObject实现为,forwardingTargetForSelector:转发给新对象。新的类动态使用
class_addMethod 添加实现。 IMP指向一个C函数。

5、 集合类,nsstring,nsdictionary,nsarray 以及他们的可变版本 插入nil,超过下标越界访问崩溃

6、NSUserDefaults崩溃。 key 或者value 有一个为NULL都会崩溃的

hook NSUserDefaults 类:

objectForKey:
setObject:forKey:

方法。

7、野指针崩溃

hook

 dealloc

方法。
就是dealloc 之后,给当前实例对象通过

    objc_destructInstance(self);
    object_setClass(self, [xxxZombieSub class]);

方法,修改isa,让其所有方法,都转向重新定向的类
在新类的

 - (id)forwardingTargetForSelector:(SEL)selector 。

转发
-》

void unrecognizedSelectorZombie(ZombieSelectorHandle* self, SEL _cmd){
    // fromObject 对象内存free后,仍然会成为野指针
    NSString* message = [NSString stringWithFormat:@"unrecognizedSelectorZombie:%@   selector:%@",[self.fromObject class],NSStringFromSelector(_cmd)];
    handleCrashException(xxxCrashProtectionZombie,message,@{});
    NSLog(@"%@", message);
}


8、非主线程操作UI

hook
UIView的下面方法

(&onceToken, ^{
            NSDictionary * methods = @{@"setNeedsLayout": @"xxxSafe_setNeedsLayout",
                                       @"setNeedsDisplay": @"xxxSafe_setNeedsDisplay",
                                       @"setNeedsDisplayInRect:": @"xxxSafe_setNeedsDisplayInRect:",
                                       @"addSubview:": @"xxxSafe_addSubview:",
                                       @"removeFromSuperview:": @"xxxSafe_removeFromSuperview:",
                                       };

方法

9、其他崩溃

NSJSONSerialization

JSONObjectWithData:options:error:

JSON序列化 data是nil
HOOK NSJSONSerialization此方法。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 206,214评论 6 481
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 88,307评论 2 382
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 152,543评论 0 341
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 55,221评论 1 279
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 64,224评论 5 371
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,007评论 1 284
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,313评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,956评论 0 259
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,441评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,925评论 2 323
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,018评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,685评论 4 322
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,234评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,240评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,464评论 1 261
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,467评论 2 352
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,762评论 2 345