【iOS】SwiftUI导致iOS12及以下系统闪退问题
背景
接入了新版的FirebaseSDK之后,测试发现iOS12系统闪退了,而iOS13及以上版本没问题。
通过下载iOS12模拟器进行测试,发现iOS12模拟器启动崩溃了,里面的提示信息为
dyld: Library not loaded: /System/Library/Frameworks/SwiftUI.framework/SwiftUI
Referenced from: /Users/handsome/Library/Developer/CoreSimulator/Devices/B7DD7057-0DC7-47B0-B783-D2BA487CE81B/data/Containers/Bundle/Application/3C31E11E-3716-4176-9C0F-6B3521637D07/SDKDemo.app/SDKDemo
Reason: image not found
dyld: launch, loading dependent libraries
DYLD_FRAMEWORK_PATH=/Users/handsome/Library/Developer/Xcode/DerivedData/SDKDemo-cmgpdjdqjjgyinchqiukyuxwzynx/Build/Products/Debug-iphonesimulator
DYLD_FALLBACK_LIBRARY_PATH=/Library/Developer/CoreSimulator/Profiles/Runtimes/iOS 12.4.simruntime/Contents/Resources/RuntimeRoot/usr/lib
DYLD_ROOT_PATH=/Library/Developer/CoreSimulator/Profiles/Runtimes/iOS 12.4.simruntime/Contents/Resources/RuntimeRoot
DYLD_FALLBACK_FRAMEWORK_PATH=/Library/Developer/CoreSimulator/Profiles/Runtimes/iOS 12.4.simruntime/Contents/Resources/RuntimeRoot/System/Library/Frameworks
DYLD_INSERT_LIBRARIES=/Library/Developer/CoreSimulator/Profiles/Runtimes/iOS 12.4.simruntime/Contents/Resources/RuntimeRoot/usr/lib/libBacktraceRecording.dylib:/Library/Developer/CoreSimulator/Profiles/Runtimes/iOS 12.4.simruntime/Contents/Resources/RuntimeRoot/usr/lib/libMainThreadChecker.dylib:/Library/Developer/CoreSimulator/Profiles/Runtimes/iOS 12.4.simruntime/Contents/Resources/RuntimeRoot/Developer/Library/PrivateFrame
关键信息为 第一行
dyld: Library not loaded: /System/Library/Frameworks/SwiftUI.framework/SwiftUI
看样子是SwiftUI的问题,Google一波之后,确认SwiftUI是iOS13开始使用,iOS13以下是没有的。
如果使用了SwiftUI,那么不进行特别处理的话,在iOS12上会崩溃,哪怕你使用了#available进行处理。
因为使用了SwiftUI之后Xcode会默认导入SwiftUI.framework库。而且默认导入的framework都是Required类型。
解决方案
知道了原因之后,解决方案也就出来了,就是将SwiftUI.framework系统库改为可选。目前有两个解决方案:
方案一:
Build Phases选项卡中的 Link Binary With Libraries 里面添加 SwiftUI.framework,然后将Status改为Optional。
方案二:
Build Settings 选项卡中的 Other Linker Flags 设置里面添加 -weak_framework SwiftUI
image.png
检查配置
目前确认是跟SwiftUI.framework相关,那么该怎么去检测出的包究竟有没有将SwiftUI.framework改为可选呢?
我尝试使用 otool -L 命令将IPA里面的可执行文件的库引入进行比对。
以下是会崩溃的IPA包分析结果(部分):
/System/Library/Frameworks/SwiftUI.framework/SwiftUI (compatibility version 0.0.0, current version 0.0.0)
.......(省略)
@rpath/libswiftUIKit.dylib (compatibility version 1.0.0, current version 5504.0.0, weak)
.......(省略)
以下是修改后没问题的IPA包分析结果(部分):
/System/Library/Frameworks/SwiftUI.framework/SwiftUI (compatibility version 0.0.0, current version 0.0.0, weak)
.......(省略)
@rpath/libswiftUIKit.dylib (compatibility version 1.0.0, current version 5100.0.0, weak)
.......(省略)
/System/Library/Frameworks/SwiftUI.framework/SwiftUI (compatibility version 0.0.0, current version 0.0.0, weak)
很明显,将SwiftUI.framework改为Optional后会多个weak的显示。
所以,确实可以通过【otool -L 可执行文件路径】这个命令进行检查。SwiftUI.framework一行后面有weak代表已经将其设置成Optional,没有那就是没有设置。
参考文章
stackoverflow:dyld: Library not loaded SwiftUI when app runs on iOS 12 using @available(iOS 13.0, *)
https://github.com/firebase/firebase-ios-sdk/releases
https://firebase.google.com/support/release-notes/ios
苹果开发者论坛:dyld: Library not loaded: /System/Library/Frameworks/SwiftUI.framework/SwiftUI
掘金:Xcode version 13.2 & 13.2.1 在iOS12上 由libswift_Concurrency.dylib 引起的crash修复