NSFileWrapper,延迟加载和保存

时间:2013-10-07 15:36:28

标签: objective-c macos cocoa nsdocument nsfilewrapper

我有一个基于NSDocument的应用程序,它使用filewrappers来保存和加载其数据。 该文档可以拥有各种资源,因此我不想将所有内容加载到内存中。我可能会做一些根本错误的事情,但是一旦我更改了一个(内部)文件然后保存,我就无法读取任何尚未加载到内存中的文件。

我已将相关代码分离到一个单独的项目中以重现此行为,并得到相同的结果。基本流程如下:

  • 我从磁盘加载现有文档。主fileWrapper是一个目录filewrapper(我称之为 main ),其中包含另外两个文件包装器( sub1 sub2 )。此时,两个内部文件包装器已加载。
  • 当用户想要编辑 sub1 时,它将从磁盘加载。
  • 用户保存文档
  • 如果用户想要编辑其他文件( sub2 ),则无法加载。出现的错误:

    -[NSFileWrapper regularFileContents] tried to read the file wrapper's contents lazily but an error occurred: The file couldn’t be opened because it doesn’t exist.
    

以下是我项目中的相关代码:

这段代码可能更易于阅读: https://gist.github.com/bob-codingdutchmen/6869871

#define FileName01 @"testfile1.txt"
#define FileName02 @"testfile2.txt"

/**
 *  Only called when initializing a NEW document
 */
-(id)initWithType:(NSString *)typeName error:(NSError *__autoreleasing *)outError {
    self = [self init];
    if (self) {
        self.myWrapper = [[NSFileWrapper alloc] initDirectoryWithFileWrappers:nil];

        NSLog(@"Initializing new document...");

        NSString *testString1 = @"Lorem ipsum first sub file";
        NSString *testString2 = @"This is the second sub file with completely unrelated contents";

        NSFileWrapper *w1 = [[NSFileWrapper alloc] initRegularFileWithContents:[testString1 dataUsingEncoding:NSUTF8StringEncoding]];
        NSFileWrapper *w2 = [[NSFileWrapper alloc] initRegularFileWithContents:[testString2 dataUsingEncoding:NSUTF8StringEncoding]];

        w1.preferredFilename = FileName01;
        w2.preferredFilename = FileName02;

        [self.myWrapper addFileWrapper:w1];
        [self.myWrapper addFileWrapper:w2];

    }
    return self;
}

-(NSFileWrapper *)fileWrapperOfType:(NSString *)typeName error:(NSError *__autoreleasing *)outError {

    // This obviously wouldn't happen here normally, but it illustrates
    // how the contents of the first file would be replaced
    NSFileWrapper *w1 = [self.myWrapper.fileWrappers objectForKey:FileName01];
    [self.myWrapper removeFileWrapper:w1];

    NSFileWrapper *new1 = [[NSFileWrapper alloc] initRegularFileWithContents:[@"New file contents" dataUsingEncoding:NSUTF8StringEncoding]];
    new1.preferredFilename = FileName01;

    [self.myWrapper addFileWrapper:new1];

    return self.myWrapper;
}

-(BOOL)readFromFileWrapper:(NSFileWrapper *)fileWrapper ofType:(NSString *)typeName error:(NSError *__autoreleasing *)outError {
    self.myWrapper = fileWrapper;
    return YES;
}

- (IBAction)button1Pressed:(id)sender {
    // Read from file1 and show result in field1
    NSFileWrapper *w1 = [[self.myWrapper fileWrappers] objectForKey:FileName01];
    NSString *string1 = [[NSString alloc] initWithData:w1.regularFileContents encoding:NSUTF8StringEncoding];
    [self.field1 setStringValue:string1];
}

- (IBAction)button2Pressed:(id)sender {
    // Read from file2 and show result in field2
    NSFileWrapper *w2 = [[self.myWrapper fileWrappers] objectForKey:FileName02];
    NSString *string2 = [[NSString alloc] initWithData:w2.regularFileContents encoding:NSUTF8StringEncoding];
    [self.field2 setStringValue:string2];
}

最底层的两种方法仅用于更新UI,因此我可以看到会发生什么。

  • 要更改文件的内容,我删除现有的fileWrapper并添加一个新文件。这是我发现改变文件内容的唯一方法,以及我在其他SO答案中看到的方式。

  • 当从磁盘加载文档时,我保留了fileWrapper,以便我可以使用它(在上面的代码中称为 myWrapper

Apple文档说NSFileWrapper支持延迟加载和增量保存,所以我假设我的代码有一些我看不到的根本缺陷。

3 个答案:

答案 0 :(得分:1)

NSFileWrapper本质上是unix文件节点的包装器。如果移动文件,则包装器保持有效。

你似乎遇到的问题是在保存期间创建一个新的文件包装器是一个新文件夹。系统会删除您之前的包装器,包括 sub2

要实现您想要的功能,您需要更改为增量保存,即仅保存更改的部件。请参阅NSDocument中的“保存到位”。

答案 1 :(得分:1)

-fileWrapperOfType:error:方法中,尝试构建一个新文件包装器,其中包含已更改成员的新内容,并引用未更改成员的旧文件包装器。

答案 2 :(得分:0)

遵循addFileWrapper的文档:你向它添加一个子(子目录),意味着

目录/

  1. addfileWrapper:FILENAME1 目录/文件名1 /

  2. addfileWrapper:文件名2 目录/文件名1 /文件名2。 该文件不存在。

  3. 你必须使用 addRegularFileWithContents:preferredFilename: 代替。