Application tried to present <_SFAppPasswordSavingViewController> 崩溃治理

一. 背景

线上捕获到了关于_SFAppPasswordSavingViewController的相关崩溃。具体崩溃堆栈如下:

这个崩溃出现在钱包提现输入密码页面。

二. 原因分析

从崩溃原因分析:

vbnet 复制代码
Application Specific Information:
*** Terminating app due to uncaught exception 'NSInvalidArgumentException', reason: 'Application tried to present modally a view controller <_SFAppPasswordSavingViewController: 0x120d60030> that is already being presented by <UIKeyboardHiddenViewController_Save: 0x123849e30>.
UserInfo:(null)'

是因为Application尝试去present存储密码_SFAppPasswordSavingViewController页面,但这个_SFAppPasswordSavingViewController已经present过,导致出现这个崩溃。而从产生该崩溃系统分析来看,这个崩溃基本发生在iOS16及相关系统,从目前来看应该是系统原因,主要出现键盘弹起自动密码填充界面。

我们从苹果论坛上面也能看到关于这个crash的相关反馈:

developer.apple.com/forums/thre...

里面有人提到针对这个崩溃的解决方法如下:

github.com/hackiftekha...

也就是hookUIViewControllerpresentViewController:animated:completion:方法,然后在hook方法里面添加判断当前的VCpresentedViewController是否有值,如果为nil,表示还没present过,直接调用``presentViewController:animated:completion:,如果有值,表示当前VC已经present过了,直接return`。

scss 复制代码
#import "UIViewController+PresentKBHook.h"

@implementation UIViewController (PresentKBHook)

+ (void)load {
    [UIViewController ff_swizzleInstanceMethodWithSrcClass:[UIViewController class] srcSel:@selector(presentViewController:animated:completion:) swizzledSel:@selector(cr_presentViewController:animated:completion:)];
}

- (void)cr_presentViewController:(UIViewController *)viewControllerToPresent animated:(BOOL)flag completion:(void (^)(void))completion {
    if (self.presentedViewController) {
        NSLog(@"[present devil]self=%@,toPresent=%@",self.description, viewControllerToPresent.description);
    }else{
        [self cr_presentViewController:viewControllerToPresent animated:flag completion:completion];
    }
}

/**
 Implementation of exchanging two object methods
 */
+ (void)ff_swizzleInstanceMethodWithSrcClass:(Class)srcClass
                                      srcSel:(SEL)srcSel
                                 swizzledSel:(SEL)swizzledSel{
    
    Method srcMethod = class_getInstanceMethod(srcClass, srcSel);
    Method swizzledMethod = class_getInstanceMethod(srcClass, swizzledSel);
    if (!srcClass || !srcMethod || !swizzledMethod) return;
    
    //Add a layer of protection measures. If the method is added successfully, it means that the method does not exist in this class, but exists in the parent class, and the method of the parent class cannot be exchanged, otherwise the method will crash when the parent class object calls the method; Adding failure indicates that the method exists in this class
    BOOL addMethod = class_addMethod(srcClass, srcSel, method_getImplementation(swizzledMethod), method_getTypeEncoding(swizzledMethod));
    if (addMethod){
        //After the IMP is successfully implemented by adding methods, the original implementation will be replaced with the swizzledMethod method to achieve method exchange without affecting the implementation of the parent method
        class_replaceMethod(srcClass, swizzledSel, method_getImplementation(srcMethod), method_getTypeEncoding(srcMethod));
    }else{
        //Adding failed, calling the implementation of the two interactive methods
        method_exchangeImplementations(srcMethod, swizzledMethod);
    }
}

@end

三. 解决方案

  • 添加降级开关、添加系统版本判断、只在iOS 16相关系统启用该崩溃治理。
swift 复制代码
    /// 键盘崩溃修复策略
    public struct FJFPresentCrashFixPolicy: HandyJSON {
        /// presnet事件 崩溃修复
        public var viewControllerPresentCrashFixEnable: Bool = false
        /// 开始 修复 版本
        public var startFixVersion: String = "16.0"
        /// 结束 修复 版本
        public var endFixVersion: String = "17.0"
        
        public init() {}
    }
    
    /// 是否为需要修复的系统版本
     public static func isNeedToFixSystemVersionCrash(startVersion: String, endVersion: String) -> Bool {
        let curVersion = UIDevice.current.systemVersion
        if curVersion.compare(startVersion, options: .numeric) == .orderedDescending,
           curVersion.compare(endVersion, options: .numeric) == .orderedAscending {
            return true
        }
        return false
    }
}
相关推荐
空中海18 小时前
第七章:vue工程化与构建工具
前端·javascript·vue.js
zhensherlock19 小时前
Protocol Launcher 系列:Trello 看板管理的协议自动化
前端·javascript·typescript·node.js·自动化·github·js
zhuà!19 小时前
element的el-form提交校验没反应问题
前端·elementui
龙猫里的小梅啊19 小时前
CSS(一)CSS基础语法与样式引入
前端·css
小码哥_常19 小时前
从0到1,开启Android音视频开发之旅
前端
渔舟小调19 小时前
P19 | 前端加密通信层 pikachuNetwork.js 完整实现
开发语言·前端·javascript
qq_120840937119 小时前
Three.js 工程向:Draw Call 预算治理与渲染批处理实践
前端·javascript
不会聊天真君6471 天前
JavaScript基础语法(Web前端开发笔记第三期)
前端·javascript·笔记
IT_陈寒1 天前
SpringBoot自动配置这破玩意儿又坑我一次
前端·人工智能·后端