我希望这与我在这里使用可变数组这一事实无关,但这让我感到困惑,所以如果是这样的话我也不会感到惊讶。
背景:
我制作了一个小型数据库,它本质上是一个包含自定义对象的 NSMutableArray,我们可以将其称为 recordObjects。我设置了数组:
database = [[NSMutableArray alloc] init];
我的自定义对象,称为“recordObject”,包含以下变量和初始化:
NSString *name;
int anInt;
Bool aBool;
我还合成了方法,这样我就可以像这样调用:
aString = [[database objectAtIndex:someIndex] name];
并向我的 Controller 类添加了方法来添加、删除和选择要显示的单个记录。到目前为止,一切正常,完全符合预期。
接下来,我设置了我的 recordObject 类(NSObject 的子类)以使用 NSCoder(通过包含在 @interface 指令中,并在实现文件中添加了以下自定义编码器和解码器方法:
-(void) encodeWithCoder: (NSCoder *) encoder {
[encoder encodeObject: name forKey: @"recordName"];
[encoder encodeInt: anInt forKey: @"recordInteger"];
[encoder encodeBool: aBool forKey: @"recordBool"];
}
-(id) initWithCoder: (NSCoder *) decoder {
name = [decoder decodeObjectForKey: @"recordName"];
anInt = [decoder decodeIntForKey: @"recordInteger"];
aBool = [decoder decodeBoolForKey: @"recordBool"];
}
为了写入文件,我使用了以下内容:
[NSKeyedArchiver archiveRootObject:database toFile:myPath];
当我运行该程序时,一切似乎都正常工作。为数组中的每条记录调用编码器方法,并将文件写入磁盘。用 TextEdit 打开文件显示数据在那里(虽然我大部分都无法理解。)
问题:
这是我遇到障碍的地方。
我添加了以下代码以将文件加载到我的数据库数组中:
database = [NSKeyedUnarchiver unarchiveObjectWithFile:myPath];
当我再次运行程序时,这次加载数据库,它似乎可以正常工作。我的第一个测试是使用 NSMutableArray 计数方法:
x = [database count];
结果是 X 填充了文件中正确数量的记录。如果我保存数据库时有5条记录,则在下次执行程序加载数据库后X设置为5。
现在,这是一个大问题:
如果我尝试使用我的任何访问器方法,程序就会崩溃。例如,如果我在加载数据库后尝试使用以下内容:
aString = [[database objectAtIndex:someIndex] name];
程序崩溃并在控制台中返回以下错误:
Program received signal: “EXC_BAD_ACCESS”.
sharedlibrary apply-load-rules all
我的解释是,出于某种原因,数据没有被正确地加载和初始化到数据库数组中,但就我的生活而言,我无法弄清楚这里哪里出了问题。
附带说明一下,我实现的所有内容都来自 Stephen G. Kochan 的书“Programming in Objective-C”
任何想法将不胜感激!
最佳答案
您的代码存在一些问题。
您的initWithCoder:
方法没有完全实现。您必须调用 [super init]
并返回 self
。您还必须复制
或保留
字符串对象,否则它将被自动释放:
- (id)initWithCoder:(NSCoder *)decoder
{
self = [super init];
if(self)
{
name = [[decoder decodeObjectForKey: @"recordName"] copy];
anInt = [decoder decodeIntForKey: @"recordInteger"];
aBool = [decoder decodeBoolForKey: @"recordBool"];
}
return self;
}
另一个问题是这一行:
database = [NSKeyedUnarchiver unarchiveObjectWithFile:myPath];
这很好,除了两件事:
- 你没有持有对象的引用,所以它会是 自动发布。
NSKeyedArchiver
返回一个不可变对象(immutable对象),在本例中是一个NSArray
而不是NSMutableArray
。
你需要这样做:
database = [[NSKeyedUnarchiver unarchiveObjectWithFile:myPath] mutableCopy];
这将保留对象(因为它是一个副本)并使对象成为 NSMutableArray
。
关于objective-c - NSKeyedArchiver 和 NSKeyedUnarchiver 与 NSMutableArray,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/10391803/