NSFileWrapper,延迟加载和保存

Bob*_*ork 13 macos cocoa objective-c nsdocument nsfilewrapper

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

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

  • 我从磁盘加载现有文档.主fileWrapper是一个目录filewrapper(我称之为main),包含另外两个filewrappers(sub1sub2).此时加载两个内部文件包装器.
  • 当用户想要编辑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.
    
    Run Code Online (Sandbox Code Playgroud)

这是我项目中的相关代码:

此代码可能更容易阅读这个要点: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];
}
Run Code Online (Sandbox Code Playgroud)

最后两种方法仅用于更新UI,以便我可以看到会发生什么.

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

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

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

Coc*_*ics 1

NSFileWrapper 本质上是 unix 文件节点的包装器。如果文件被移动,包装器仍然有效。

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

要实现您想要的效果,您需要更改为增量保存,即仅将更改的部分保存到位。请参阅 NSDocument 中的“就地保存”。