我在iOS中创建了套接字连接机制。它运行得很好。但它不时(约1%)崩溃。这是崩溃日志,你知道为什么会发生吗。
Thread 0 Crashed:
0 libobjc.A.dylib 0x3b01b5b0 _objc_msgSend + 16
1 CoreFoundation 0x332957cf _signalEventSync + 75
2 CoreFoundation 0x3329b623 _cfstream_solo_signalEventSync + 75
3 CoreFoundation 0x33295507 _CFStreamSignalEvent + 327
4 CFNetwork 0x32ffa6ff CoreWriteStreamCFStreamSupport::coreStreamWriteEvent(__CoreWriteStream*, unsigned long) + 99
5 CFNetwork 0x32ffa0b5 CoreWriteStreamClient::coreStreamEventsAvailable(unsigned long) + 37
6 CFNetwork 0x32ffb365 CoreStreamBase::_callClientNow() + 45
7 CFNetwork 0x32ffb0f9 CoreStreamBase::_streamSetEventAndScheduleDelivery(unsigned long, unsigned char) + 89
8 CFNetwork 0x32ffb4ff CoreStreamBase::_streamInterface_SignalEvent(unsigned long, CFStreamError const*) + 35
9 CFNetwork 0x32f69b57 SocketStream::socketCallback(__CFSocket*, unsigned long, __CFData const*, void const*) + 135
10 CFNetwork 0x32f69ab3 SocketStream::_SocketCallBack_stream(__CFSocket*, unsigned long, __CFData const*, void const*, void*) + 75
11 CoreFoundation 0x332cfd81 __CFSocketPerformV0 + 385
12 CoreFoundation 0x332cd683 __CFRUNLOOP_IS_CALLING_OUT_TO_A_SOURCE0_PERFORM_FUNCTION__ + 15
13 CoreFoundation 0x332ccf7f __CFRunLoopDoSources0 + 363
14 CoreFoundation 0x332cbcb7 __CFRunLoopRun + 647
15 CoreFoundation 0x3323eebd _CFRunLoopRunSpecific + 357
16 CoreFoundation 0x3323ed49 _CFRunLoopRunInMode + 105
17 GraphicsServices 0x36e172eb _GSEventRunModal + 75
18 UIKit 0x35154301 _UIApplicationMain + 1121
19 Okey101Plus 0x000e19ab main (main.m:16)
看起来套接字类正试图向您的委托发送消息,而该委托已成为僵尸。
NSStream类不像iOS/OSX中的大多数其他网络类那样不保留其委托,所以如果你不在某个地方保留它,它就会消失。然后,当该类试图告诉您的类流有数据可供读取或空间可供写入时,您将在objc_msgSend中遇到崩溃,如下图所示。
如果您确实保留了它,请确保在删除保留的委托之前完全删除流,并小心在哪个线程上删除内容,以确保在释放委托后不会在主运行循环上触发已经计划好的回调。
或者您未能实现所需的委托方法。然而,这不符合"很少发生"的说法。:-)
我真的不太确定,但根据你的报告,在套接字上写东西似乎失败了-
CoreWriteStreamCFStreamSupport::coreStreamWriteEvent(__CoreWriteStream*, unsigned long) + 99
可能有一种方法可以避免这个错误。CFNetworks框架有一个很好的功能,叫做CFNetDiagnostics参考。请仔细看一下苹果文档。
来自Apple文档:在许多基于网络的应用程序中,可能会发生与应用程序无关的基于网络的错误。(例如,他们的意思可能是:如果接口不可用,当网络连接丢失时,套接字上可能会出现读取错误)然而,大多数用户可能不知道应用程序失败的原因。CFNetDiagnostics API使您能够快速轻松地帮助用户解决网络问题,而您只需在端上做少量工作。
要通过网络诊断助手诊断问题,请调用CFNetDiagnosticDiagnoseProbemInteractivey函数并传递网络诊断参考。清单6-1展示了如何将CFNetDiagnostics与在运行循环中实现的流一起使用。
case kCFStreamEventErrorOccurred:
CFNetDiagnosticRef diagRef =
CFNetDiagnosticCreateWithStreams(NULL, stream, NULL);
(void)CFNetDiagnosticDiagnoseProblemInteractively(diagRef);
CFStreamError error = CFReadStreamGetError(stream);
reportError(error);
CFReadStreamClose(stream);
CFRelease(stream);
break;
您可以在此处阅读更多信息-使用网络诊断,CFNetDiagnostics参考