JSONDecoder 在 TestFlight / AppStore 上崩溃
Posted
技术标签:
【中文标题】JSONDecoder 在 TestFlight / AppStore 上崩溃【英文标题】:JSONDecoder crashing on TestFlight / AppStore 【发布时间】:2020-02-05 22:08:54 【问题描述】:我正在开发我的 Apple Watch 应用。我的大部分组件都使用 JSON 并使用 Swift Codable 协议来解码 JSON。 现在我想准备我的版本,但我收到一个 Beta 测试人员的消息,说应用程序在他的 Series 3 上总是崩溃。
在我的 Series 3 上安装 TestFlight 构建后,我可以重现该问题。 来自 TestFlight 的崩溃日志和从我的 iPhone 中提取的崩溃日志表明在 JSON 被解码时发生了崩溃。
我有相当复杂的 JSON 结构,但自上次更新以来我没有对它们进行任何更改。此外,解码 JSON 在 Xcode 构建(也适用于系列 3)和较新的 Apple Watch(系列 4 + 5)上运行良好。我正在努力找出可能导致问题的原因。 我在 Apple 开发者论坛中发现了一个类似问题的帖子,但没有解决方案:https://forums.developer.apple.com/thread/124627
这是我的崩溃日志的 Stacktrace 的一部分:
Date/Time: 2020-02-05 08:37:42.9924 +0100
Launch Time: 2020-02-05 08:37:40.0000 +0100
OS Version: Watch OS 6.1.2 (17S5792a)
Release Type: User
Baseband Version: 4.40.02
Report Version: 104
Exception Type: EXC_CRASH (SIGABRT)
Exception Codes: 0x0000000000000000, 0x0000000000000000
Exception Note: EXC_CORPSE_NOTIFY
Triggered by Thread: 0
Application Specific Information:
abort() called
Thread 0 name: Dispatch queue: com.apple.main-thread
Thread 0 Crashed:
0 libsystem_kernel.dylib 0x4d9aefc8 __pthread_kill + 8
1 libsystem_pthread.dylib 0x4da260be pthread_kill + 108
2 libsystem_c.dylib 0x4d9383a4 abort + 84
3 libswiftCore.dylib 0x6cffbade swift_vasprintf+ 2341598 (char**, char const*, char*) + 0
4 libswiftCore.dylib 0x6d00932c swift::TargetMetadata<swift::InProcess>::getGenericArgs+ 2396972 () const + 0
5 libswiftCore.dylib 0x6d00508c _swift_initClassMetadataImpl+ 2379916 (swift::TargetClassMetadata<swift::InProcess>*, swift::ClassLayoutFlags, unsigned long, swift::TypeLayout const* const*, unsigned long*, bool) + 28
6 libswiftCore.dylib 0x6d005c86 swift_initClassMetadata2 + 22
7 libswiftCore.dylib 0x6cfdc6fc type metadata completion function for _KeyedDecodingContainerBox + 50
8 libswiftCore.dylib 0x6d00a6c2 swift::MetadataCacheEntryBase<(anonymous namespace)::GenericCacheEntry, void const*>::doInitialization+ 2401986 (swift::ConcurrencyControl&, swift::MetadataCompletionQueueEntry*, swift::MetadataRequest) + 178
9 libswiftCore.dylib 0x6d003a0c swift_getGenericMetadata + 1124
10 libswiftCore.dylib 0x6cdf4210 KeyedDecodingContainer.init<A>+ 213520 (_:) + 40
11 libswiftFoundation.dylib 0x6d156e0a __JSONDecoder.container<A>+ 323082 (keyedBy:) + 678
12 libswiftFoundation.dylib 0x6d1a3a1c protocol witness for Decoder.container<A>+ 637468 (keyedBy:) in conformance __JSONDecoder + 18
13 libswiftFoundation.dylib 0x6d1592b6 protocol witness for Decoder.container<A>+ 332470 (keyedBy:) in conformance __JSONDecoder + 32
14 libswiftCore.dylib 0x6cfdb1cc dispatch thunk of Decoder.container<A>+ 2208204 (keyedBy:) + 24
15 WatchSonos WatchKit Extension 0x00440bf2 0x358000 + 953330
16 WatchSonos WatchKit Extension 0x0043d08c 0x358000 + 938124
17 libswiftCore.dylib 0x6cfdb156 dispatch thunk of Decodable.init+ 2208086 (from:) + 12
18 libswiftFoundation.dylib 0x6d16fbee __JSONDecoder.unbox_+ 424942 (_:as:) + 4014
19 libswiftFoundation.dylib 0x6d156816 JSONDecoder.decode<A>+ 321558 (_:from:) + 642
20 libswiftFoundation.dylib 0x6d22af98 dispatch thunk of JSONDecoder.decode<A>+ 1191832 (_:from:) + 28
21 WatchSonos WatchKit Extension 0x0045db0c 0x358000 + 1071884
22 WatchSonos WatchKit Extension 0x00461cce 0x358000 + 1088718
23 WatchSonos WatchKit Extension 0x00461bde 0x358000 + 1088478
24 WatchSonos WatchKit Extension 0x00420ffa 0x358000 + 823290
25 WatchSonos WatchKit Extension 0x00425914 0x358000 + 842004
26 libdispatch.dylib 0x4d851846 _dispatch_call_block_and_release + 10
27 libdispatch.dylib 0x4d8528b8 _dispatch_client_callout + 6
28 libdispatch.dylib 0x4d85c6b0 _dispatch_main_queue_callback_4CF + 868
29 CoreFoundation 0x4dcdbbaa __CFRUNLOOP_IS_SERVICING_THE_MAIN_DISPATCH_QUEUE__ + 10
30 CoreFoundation 0x4dcd7c00 __CFRunLoopRun + 1832
31 CoreFoundation 0x4dcd723a CFRunLoopRunSpecific + 370
32 GraphicsServices 0x50aebcd0 GSEventRunModal + 96
33 UIKitCore 0x65c91580 UIApplicationMain + 1730
34 libxpc.dylib 0x4da73c80 _xpc_objc_main.cold.3 + 152
35 libxpc.dylib 0x4da649b0 _xpc_objc_main + 184
36 libxpc.dylib 0x4da668c4 xpc_main + 110
37 Foundation 0x4e56b0c6 +[NSXPCListener serviceListener] + 0
38 PlugInKit 0x556d726c 0x556c5000 + 74348
39 WatchKit 0x5cfe9afe WKExtensionMain + 62
40 WatchKit 0x5d09f020 main + 10
这是代码中崩溃的位置:
required public init(from decoder: Decoder) throws
let values = try decoder.container(keyedBy: CodingKeys.self)
self.id = try values.decode(String.self, forKey: .id)
do
if values.contains(.groups)
let groups = try values.decode(Dictionary<String, SonosGroup>.self, forKey: .groups)
self.groups = groups
if values.contains(.players)
let players = try values.decode(Array<SonosPlayer>.self, forKey: .players)
self.players = players
catch let error
print("Error decoding groups \(error)")
对我来说,这看起来像是一个 Apple 错误,我为此提交了反馈,但这不是下一个版本的解决方案。
有谁知道 TestFlight 构建和 Xcode 发布构建之间的区别是什么?
我尝试设置不同的优化级别,禁用位码(Apple Watch 应用程序不允许),然后我开始用旧的 JSONSerialization
替换 Codable。
如果有人知道更多,谢谢。
亲切的问候 亚历克斯
【问题讨论】:
函数的前两行在 do...catch 块之外。我会将它们移动到块中,为错误添加一些日志记录到设备日志中(不仅仅是仅在调试控制台中显示的打印语句),将构建推送到 TestFlight,然后查看日志。您还可以添加一个专门转换为 DecodingError 的 catch,因为这可能有助于找出正在发生的错误类型。 嗨,马克,感谢您的评论,但遗憾的是这不是问题所在。它没有抛出任何错误。这只是一个纯粹的崩溃,否则 id 是强制性的,所以它应该在这里抛出一个错误。错误在调用 JSONDecoder 的函数上被捕获。 我注意到你的初始化是public
。您是否检查过 CodingKeys 的可访问性?我最近遇到了一个问题,标记为 private
的 CodingKeys 导致了一些意外行为。没有抛出错误,但默默地失败了。反正只是一个想法。
【参考方案1】:
我在 swift 包中遇到了与 Encodable 和 Decodable 非常相似的问题。在我的应用程序目标中设置 DEAD_CODE_STRIPPING = NO
修复了它。这似乎是这个错误的一个奇怪的遗留物:
https://bugs.swift.org/browse/SR-11564?focusedCommentId=51285&page=com.atlassian.jira.plugin.system.issuetabpanels%3Acomment-tabpanel#comment-51285
【讨论】:
以上是关于JSONDecoder 在 TestFlight / AppStore 上崩溃的主要内容,如果未能解决你的问题,请参考以下文章
Swift - JSONDecoder & Combine 框架问题