weak_clear_no_lock 报错处理

项目上线之后,通过firbase追踪发现有一个异常报错,闪退率惊人,起初我们以为是firebase同步了WKWebView的异常信息导致的,但后来在本人手机上复测发现实际上不是这个问题,是真正Bug存在。

Crashed: com.apple.main-thread
0  libsystem_kernel.dylib         0x1af90d858 __abort_with_payload + 8
1  libsystem_kernel.dylib         0x1af911ed8 abort_with_payload_wrapper_internal + 104
2  libsystem_kernel.dylib         0x1af911e70 abort_with_payload_wrapper_internal + 30
3  libobjc.A.dylib                0x1af85a3cc _objc_fatalv(unsigned long long, unsigned long long, char const*, char*) + 108
4  libobjc.A.dylib                0x1af85a360 _objc_fatalv(unsigned long long, unsigned long long, char const*, char*) + 30
5  libobjc.A.dylib                0x1af856530 weak_clear_no_lock + 322
6  libobjc.A.dylib                0x1af857f70 objc_storeWeak + 336
7  WebKit                         0x1b743d5b0 WebKit::UIDelegate::setDelegate(id) + 32
8  WebKit                         0x1b71d2acc -[WKWebView setUIDelegate:] + 48
9  XXX                            0x1053e343c -[XXXWebViewController webView] + 426 (XXXWebViewController.m:426)
10 XXX                            0x1053e3318 -[XXXWebViewController dealloc] + 416 (XYFWebViewController.m:416)
11 XXX                            0x1053d97f4 -[XXXManager viewPageChangeWithTarget:actionItem:complete:] + 138 (XXXManager.m:138)
12 XXX                            0x1053d9530 __54-[XXXManager startLoanWithTarget:actionItem:complete:]_block_invoke.115 + 120 (XXXManager.m:120)
13 XXX                            0x1053e6740 __68-[XXXNetClient getUserObjectAfterloginSuccessedWithTarget:callback:]_block_invoke + 232 (XXXNetClient.m:232)
14 libdispatch.dylib              0x1af7c6ec4 _dispatch_call_block_and_release + 32
15 libdispatch.dylib              0x1af7c833c _dispatch_client_callout + 20

看日志是不是以为WKWebView的问题,实际上经过调试发现,是因为当前controller释放的时候,又在dealloc调用了WKWebView设置UIDelegate的方法,但是因为当前controller正在被释放中,所以产生了weak_clear_no_lock报错。报错打印信息如下:

Cannot form weak reference to instance (0x160f6f890) of class XXXWebViewController. It is possible that this object was over-released, or is in the process of deallocation.

场景:登录页面是present出来的,登录完毕后,页面dismiss消失,然后跳转到webView页面,但线上的原因是因为线回调了登录成功block去跳转webview,然后才dismiss页面,由于登录页面没有导航,所以无法跳转webview页面,但是创建了webView页面的controller,dismiss后,刚才创建的webView的controller就要被dealloc,但是在dealloc的时候,在webview的懒加载代理里又设置为代理。此时便会报错。 

//MARK: -- Remove Delegate
- (void)dealloc {
    [self.webView removeObserver:self forKeyPath:@"estimatedProgress"];
    [self.webView removeObserver:self forKeyPath:@"title"];
}
- (WKWebView *)webView {
    if (!_webView) {
        WKWebViewConfiguration *config = [[WKWebViewConfiguration alloc] init];
        _webView = [[WKWebView alloc] initWithFrame:CGRectZero configuration:config];
        _webView.UIDelegate = self;
        _webView.navigationDelegate = self;
        //  设置UserAgent
        [_webView evaluateJavaScript:@"navigator.userAgent" completionHandler:^(id result, NSError *error) {
            NSString *userAgent = result;
            NSString *preStr = [NSString stringWithFormat:@"%@-iOS", [XXXNetClient sharedClient].appname?:@""];
            if (![userAgent hasPrefix:preStr]) {
                NSString *customUserAgent = [preStr stringByAppendingString:userAgent];
                NSDictionary *dictionary = [NSDictionary dictionaryWithObjectsAndKeys:customUserAgent, @"UserAgent", nil];
                [[NSUserDefaults standardUserDefaults] registerDefaults:dictionary];
                [[NSUserDefaults standardUserDefaults] synchronize];
            }
        }];
    }
    return _webView;
}

weak_clear_no_lock是内存管理时释放weak指针清理资源时用到的函数,我们找来它的源码如下:

id 
weak_register_no_lock(weak_table_t *weak_table, id referent_id, id *referrer_id)
{
    objc_object *referent = (objc_object *)referent_id;
    objc_object **referrer = (objc_object **)referrer_id;

    if (!referent  ||  referent->isTaggedPointer()) return referent_id;

    // ensure that the referenced object is viable
    bool deallocating;
    if (!referent->ISA()->hasCustomRR()) {
        deallocating = referent->rootIsDeallocating();
    }
    else {
        BOOL (*allowsWeakReference)(objc_object *, SEL) = 
            (BOOL(*)(objc_object *, SEL))
            object_getMethodImplementation((id)referent, 
                                           SEL_allowsWeakReference);
        if ((IMP)allowsWeakReference == _objc_msgForward) {
            return nil;
        }
        deallocating =
            ! (*allowsWeakReference)(referent, SEL_allowsWeakReference);
    }

    if (deallocating) {
        _objc_fatal("Cannot form weak reference to instance (%p) of "
                    "class %s. It is possible that this object was "
                    "over-released, or is in the process of deallocation.",
                    (void*)referent, object_getClassName((id)referent));
    }

    // now remember it and where it is being stored
    weak_entry_t *entry;
    if ((entry = weak_entry_for_referent(weak_table, referent))) {
        append_referrer(entry, referrer);
    } 
    else {
        weak_entry_t new_entry;
        new_entry.referent = referent;
        new_entry.out_of_line = 0;
        new_entry.inline_referrers[0] = referrer;
        for (size_t i = 1; i < WEAK_INLINE_COUNT; i++) {
            new_entry.inline_referrers[i] = nil;
        }
        
        weak_grow_maybe(weak_table);
        weak_entry_insert(weak_table, &new_entry);
    }

    // Do not set *referrer. objc_storeWeak() requires that the 
    // value not change.

    return referent_id;
}

runtime 是通过检查引用计数的个数来判断对象是否在 deallocting, 然后通过deallocting来使得程序报错。问题就是这样,涉及到的Weak的实现方式可以自行了解。

解决方案1:登录页面dismiss之后再回调登录成功的block

解决方案2:   webview的controller在dealloc的时候不使用点语法,可以改为以下方式,以避免程序crash

- (void)dealloc {
    if (_webView) {
        [_webView removeObserver:self forKeyPath:@"estimatedProgress"];
        [_webView removeObserver:self forKeyPath:@"title"];
    }
}

如有错误,烦请指正!

你可能感兴趣的:(iOS技术问题)