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
    }
}
相关推荐
mCell13 小时前
JavaScript 的多线程能力:Worker
前端·javascript·浏览器
超级无敌攻城狮14 小时前
3 分钟学会!波浪文字动画超详细教程,从 0 到 1 实现「思考中 / 加载中」高级效果
前端
excel15 小时前
用 TensorFlow.js Node 实现猫图像识别(教学版逐步分解)
前端
gnip15 小时前
JavaScript事件流
前端·javascript
赵得C15 小时前
【前端技巧】Element Table 列标题如何优雅添加 Tooltip 提示?
前端·elementui·vue·table组件
wow_DG16 小时前
【Vue2 ✨】Vue2 入门之旅 · 进阶篇(一):响应式原理
前端·javascript·vue.js
weixin_4569042716 小时前
UserManagement.vue和Profile.vue详细解释
前端·javascript·vue.js
资深前端之路16 小时前
react 面试题 react 有什么特点?
前端·react.js·面试·前端框架
aaaweiaaaaaa16 小时前
HTML和CSS学习
前端·css·学习·html
秋秋小事16 小时前
React Hooks useContext
前端·javascript·react.js