gpt4 book ai didi

ios - RestKit 核心数据 NSError dealloc 崩溃

转载 作者:可可西里 更新时间:2023-11-01 05:44:22 27 4
gpt4 key购买 nike

试图弄清我在生产构建中遇到的问题的根源,最终能够在测试时重现它。使用 RestKit v0.23.1,在使用以下代码执行 RKManagedObjectRequestOperation 时(插入仪器时)我得到“Objective-C 消息已发送到已释放的‘NSError’对象(僵尸)”,并且每次有对象时应用程序都会崩溃响应 JSON - 如果响应类似于“objects = ();”没有崩溃 - 所以我猜它在 RestKit/Core Data 映射或存储中的某个地方?

    RKManagedObjectRequestOperation *objectRequestOperation = [_objectManager managedObjectRequestOperationWithRequest:request managedObjectContext:_objectManager.managedObjectStore.mainQueueManagedObjectContext success:^(RKObjectRequestOperation *operation, RKMappingResult *mappingResult) {
DDLogInfo(@"INSIDE SUCCESS BLOCK");
} failure:^(RKObjectRequestOperation *operation, NSError *error) {
DDLogInfo(@"INSIDE ERROR BLOCK");
}];

[objectRequestOperation setWillMapDeserializedResponseBlock:^id(id deserializedResponseBody) {
DDLogInfo(@"Response JSON: %@", deserializedResponseBody);

return deserializedResponseBody;
}];

objectRequestOperation.savesToPersistentStore = YES;
[objectRequestOperation start];

原始 JSON 已正确记录在 setWillMapDeserializedResponseBlock 中,但从未到达成功和错误 block 中的日志。这是我从 crashlytics 返回的堆栈跟踪:

Thread : Crashed: NSOperationQueue Serial Queue
0 libobjc.A.dylib 0x37dd4626 objc_msgSend + 5
1 Foundation 0x2df5802d -[NSError dealloc] + 60
2 libobjc.A.dylib 0x37dd9b6b objc_object::sidetable_release(bool) + 174
3 libobjc.A.dylib 0x37dda0d3 (anonymous namespace)::AutoreleasePoolPage::pop(void*) + 358
4 CoreFoundation 0x2d569501 _CFAutoreleasePoolPop + 16
5 Foundation 0x2df69999 -[__NSOperationInternal _start:] + 1064
6 Foundation 0x2e00d745 __NSOQSchedule_f + 60
7 libdispatch.dylib 0x382b8cbd _dispatch_queue_drain + 488
8 libdispatch.dylib 0x382b5c6f _dispatch_queue_invoke + 42
9 libdispatch.dylib 0x382b95f1 _dispatch_root_queue_drain + 76
10 libdispatch.dylib 0x382b98dd _dispatch_worker_thread2 + 56
11 libsystem_pthread.dylib 0x383e4c17 _pthread_wqthread + 298

最佳答案

这不是 RestKit 的问题。我经常看到这个问题,实际上看起来过度释放实际上发生在 Apple 的代码中。当您尝试保存到 Core Data 存储但失败时,就会出现问题。 Core Data 报告错误,但错误处理不当。

我遇到过一些导致保存失败的情况,这就是我修复它们的方法:

由于数据保护 API,数据存储无法访问。

要么忙着等待,让你的应用无法像这样启动:

while(![[UIApplication sharedApplication] isProtectedDataAvailable]) {
[[NSRunLoop currentRunLoop] runUntilDate:[NSDate dateWithTimeIntervalSinceNow:0.5f]];
}

或者如果您商店中的数据不像这样敏感,则禁用保护:

[_coordinator addPersistentStoreWithType:NSSQLiteStoreType
configuration:nil
URL:url
options:@{NSPersistentStoreFileProtectionKey:NSFileProtectionNone}
error:&error];

重要的是在您可以访问文件之前不要尝试保存。如果您可以重新构建代码以防止在无法访问数据库时访问数据库,那也很好。您可以使用数据保护 API 应用程序委托(delegate)方法来触发该机制。

数据存储已损坏 - 此处最好的做法是删除存储并重新开始。这是直接使用 sqlite 库检测损坏存储的好方法。

    #import <sqlite3.h>

sqlite3 *dbConnection;
if (sqlite3_open([[url absoluteString] UTF8String], &dbConnection) != SQLITE_OK) {
NSLog(@"[SQLITE] Unable to open database!");
}
sqlite3_stmt *statement = nil;
sqlite3_prepare_v2(dbConnection, "PRAGMA quick_check;", -1, &statement, NULL);
NSString *result = nil;
while (sqlite3_step(statement) == SQLITE_ROW) {
for (int i=0; i<sqlite3_column_count(statement); i++) {
int colType = sqlite3_column_type(statement, i);
if (colType == SQLITE_TEXT) {
const unsigned char *col = sqlite3_column_text(statement, i);
result = [NSString stringWithFormat:@"%s", col];
} else {
NSLog(@"[SQLITE] UNKNOWN DATATYPE");
}
}
}
sqlite3_close(dbConnection);

这会运行一个 sqlite PRAGMA 查询来执行完整性检查。我使用 quick_check,但如果您愿意等待额外的时间,您也可以使用 integrity_check。您可以使用 [result isEqualToString:@"ok"]

判断事情是否正常

关于ios - RestKit 核心数据 NSError dealloc 崩溃,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/23574607/

27 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com