NSSecureCoding 为正确保存的对象返回 nil 如何保存如何加载

问题描述

我使用的是 macOS 10.15 (bigSUr)、XCode 12、objective-c 而不是 ios。

我有基于文档的应用程序。它有一个简单的对象“SHGlobalAppData”(NSObject),其中包含一个自定义类“SHSetupDataModel”(NSObject)的属性对象。

加载时,initWithCoder 为保存的值返回 nil。为什么?

这是实现:

我使用 NSSecureCoding,因此 SHSetupDataModel 和 SHGlobalAppData 都包含了相应的类方法

+ (BOOL)supportsSecureCoding { return YES;}

使用安全编码在 NSDocument 中保存

- (NSData *)dataOfType:(Nsstring *)typeName error:(NSError **)outError {
    
    NSKeyedArchiver* archiver=[[NSKeyedArchiver alloc] initRequiringSecureCoding:YES];
    [archiver encodeObject:self.appData forKey:@"appData"]; // SHSetupDataModel is a property of appData object
    //[...]
}

如何保存

说到保存,这是SHGlobalAppData的代码

- (void)encodeWithCoder:(NSCoder *)coder {

    // Other properties here

    if (_setupData){
        // Tests
        NSLog(@"%@",[_setupData className]); // returns "SHSetupDataModel"
        BOOL test = [_setupData isKindOfClass:[SHSetupDataModel class]]; // returns TRUE

        [coder encodeObject:_setupData forKey:@"setupData"];
    }
}

以上保存运行顺利。测试没问题。

如何加载

现在在加载保存的文件时,调用以下 NSDocument 方法

- (BOOL)readFromURL:(NSURL *)url ofType:(Nsstring *)typeName error:(NSError *__autoreleasing  _Nullable *)outError {

    NSData* data = [[NSData alloc] initWithContentsOfURL:url];
    NSKeyedUnarchiver* unarchiver = [[NSKeyedUnarchiver alloc] initForReadingFromData:data error:outError];
    [unarchiver setRequiresSecureCoding:YES];

    // Load appData
    SHGlobalAppData* appData = [unarchiver decodeObjectOfClass:[SHGlobalAppData class] forKey:@"appData"];
    [unarchiver finishDecoding];
    // [...]
}

这会调用 SHGlobalAppData 的 initWithCoder 方法 - 我得到一个 nil 结果

- (id)initWithCoder:(NSCoder *)coder {

    self = [super initWithCoder:coder];
    
    if (self) {
        
        if ([coder containsValueForKey:@"setupData"]){
            _setupData = [coder decodeObjectOfClass:[SHSetupDataModel class] forKey:@"setupData"];   // <---- This is nil. Why?              
        }
        // [...]
    }
}

谁能帮助我为什么返回零?或者引导我进行更有效的调试?

解决方法

数据模型本身包含未正确解码的 NSDictionary。解决方案是分析 outError,然后通过 initWithCoder 方法逐步工作。 最后我不得不在 decodeObjectWithClasses:ofKey:

中传递多个类