编辑:
通过在函数调用之前的某处添加QCAnnonce* a = [[QCAnnonce alloc] init];
,我解决了问题,所以我想我已经将类引入了运行时"。但我现在有一个警告说'" a"是未使用的,所以我可以使用运行时函数来将类引入运行时"?
我正在尝试创建一个客户端 - 服务器应用程序。使用NSKeyed(Un)Archiver将对象存档并取消存档以进行传输。我的大多数对象都没有任何问题地传输,但其中一个引发了NSInvalidUnarchiveOperationException。
在我放入initWithCoder的断点之前调用异常:。
我尝试在发送之前存档/取消存档,这很有效,所以它不应该是initWithCoder的问题。
我试图在我的服务器应用程序中创建一个客户端(不是作为我的客户端在一个单独的应用程序中),他可以解码该对象。除了他们不在同一个应用程序中之外,我无法看到我的客户端和这个新客户端之间的任何差异。
我最好的猜测是apple doc:
的这一部分例如,代理可以加载一些代码来引入该类 到运行时并返回类,或替换不同的类 宾语。如果委托返回nil,则取消归档中止和方法 引发NSInvalidUnarchiveOperationException。
但是我不知道"将类引入运行时"装置
以下是编码/解码的对象方法:
-(id)initWithCoder:(NSCoder *)aDecoder{
self = [super init];
if (self) {
a_listeAnnonces = [aDecoder decodeObjectForKey:@"Cartes"];
a_points = [aDecoder decodeIntForKey:@"Points"];
}
return self;
}
-(void)encodeWithCoder:(NSCoder *)aCoder{
[aCoder encodeObject:a_listeAnnonces forKey:@"Cartes"];
[aCoder encodeInt:a_points forKey:@"Points"];
}
以下是异常消息:
2014-06-04 11:27:34.681 myApp[3693:303] *** -[NSKeyedUnarchiver decodeObjectForKey:]: cannot decode object of class (QCAnnonce)
2014-06-04 11:27:34.794 myApp[3693:303] (
0 CoreFoundation 0x00007fff8937a25c __exceptionPreprocess + 172
1 libobjc.A.dylib 0x00007fff8e898e75 objc_exception_throw + 43
2 CoreFoundation 0x00007fff8937a10c +[NSException raise:format:] + 204
3 Foundation 0x00007fff8a8acdd9 _decodeObjectBinary + 2349
4 Foundation 0x00007fff8a8ac34d _decodeObject + 288
5 Foundation 0x00007fff8a8d15a5 +[NSKeyedUnarchiver unarchiveObjectWithData:] + 92
6 myApp 0x000000010001365f -[QNProtocolWrap performMethod:withArgumentDatas:onObject:] + 543
7 myApp 0x000000010001261b -[QNConnection methodCall:withArguments:] + 251
8 myApp 0x0000000100012b23 -[QNConnection connectionBaseDidRecieveNewData:] + 211
9 myApp 0x0000000100012b8b -[QNConnection connectionBaseDidRecieveNewData:] + 315
10 myApp 0x000000010000e38c -[QNConnectionBase readInput] + 204
11 myApp 0x000000010000e591 -[QNConnectionBase stream:handleEvent:] + 449
12 CoreFoundation 0x00007fff892ebc81 _signalEventSync + 385
13 CoreFoundation 0x00007fff892ebac8 _cfstream_solo_signalEventSync + 328
14 CoreFoundation 0x00007fff892eb93f _CFStreamSignalEvent + 623
15 CFNetwork 0x00007fff81e4401a _ZN29CoreReadStreamCFStreamSupport19coreStreamReadEventEP16__CoreReadStreamm + 102
16 CFNetwork 0x00007fff81e43f89 _ZN20CoreReadStreamClient25coreStreamEventsAvailableEm + 53
17 CFNetwork 0x00007fff81f77a65 _ZN14CoreStreamBase14_callClientNowEP16CoreStreamClient + 53
18 CFNetwork 0x00007fff81e43ca9 _ZN14CoreStreamBase34_streamSetEventAndScheduleDeliveryEmh + 183
19 CFNetwork 0x00007fff81e43a32 _ZN12SocketStream40dispatchSignalFromSocketCallbackUnlockedEP24SocketStreamSignalHolder + 74
20 CFNetwork 0x00007fff81e43160 _ZN12SocketStream14socketCallbackEP10__CFSocketmPK8__CFDataPKv + 206
21 CFNetwork 0x00007fff81e43062 _ZN12SocketStream22_SocketCallBack_streamEP10__CFSocketmPK8__CFDataPKvPv + 64
22 CoreFoundation 0x00007fff892eb107 __CFSocketPerformV0 + 855
23 CoreFoundation 0x00007fff892ab661 __CFRUNLOOP_IS_CALLING_OUT_TO_A_SOURCE0_PERFORM_FUNCTION__ + 17
24 CoreFoundation 0x00007fff8929cd12 __CFRunLoopDoSources0 + 242
25 CoreFoundation 0x00007fff8929c49f __CFRunLoopRun + 831
26 CoreFoundation 0x00007fff8929bf25 CFRunLoopRunSpecific + 309
27 HIToolbox 0x00007fff89726a0d RunCurrentEventLoopInMode + 226
28 HIToolbox 0x00007fff897267b7 ReceiveNextEventCommon + 479
29 HIToolbox 0x00007fff897265bc _BlockUntilNextEventMatchingListInModeWithFilter + 65
30 AppKit 0x00007fff82a9526e _DPSNextEvent + 1434
31 AppKit 0x00007fff82a948bb -[NSApplication nextEventMatchingMask:untilDate:inMode:dequeue:] + 122
32 AppKit 0x00007fff82a889bc -[NSApplication run] + 553
33 AppKit 0x00007fff82a737a3 NSApplicationMain + 940
34 myApp 0x0000000100001262 main + 34
35 myApp 0x0000000100001234 start + 52
36 ??? 0x0000000000000003 0x0 + 3
)
答案 0 :(得分:1)
当您尝试通过NSKeyedArchiver对其进行解码时,似乎没有将Class加载到运行时。
如果没有加载,NSKeyedArchiver就找不到类QCAnnounce了。 (这就是它无法解码的原因)
要检查这一点,您可以查看NSObject协议的+ (void)load
方法。
在将类加载到进程的地址空间后,运行时每个类发送一次此消息。
对于属于程序可执行文件一部分的类,运行时会在进程的生命周期中尽早发送加载消息。对于共享(动态加载)库中的类,运行时会在共享库加载到进程的地址空间后立即发送加载消息。
您可以通过重载+(void)load
方法并在那里设置断点来检查这一点,看看它是否已在调用unarchive之前加载到运行时。
+ (void)load {
NSLog(@"%@ loaded!", self);
}
您的修复[[QCAnnounce alloc] init]有效,因为当Runtime尝试向类发送第一条消息时,它需要调用+(void)initialize
到该类,并在您接收初始化时,在您的进程中进行类调用应该已经收到了负载。
答案 1 :(得分:1)
这里的问题是运行时没有加载静态库中存在的所有类。由于我没有直接调用该类(只有id),因此从未加载过。
我通过在构建设置中的“其他链接器标志”中添加-objC来修复它。
在Technical Q&A中找到了一些关于为什么这是一个好主意的信息:
-objC使链接器加载库中定义Objective-C类或类别的每个目标文件
还有另外两个标志(相同的Q& A):
-all_load强制链接器从它看到的每个存档加载所有目标文件,即使那些没有Objective-C代码的存档也是如此。 -force_load在Xcode 3.2及更高版本中可用。它允许更精细的谷物控制存档加载。每个-force_load选项后面都必须跟一个存档路径,并且将加载该存档中的每个目标文件。