NSManagedObject [GHUnit] 的单元测试

Posted

技术标签:

【中文标题】NSManagedObject [GHUnit] 的单元测试【英文标题】:Unit test for NSManagedObject [GHUnit] 【发布时间】:2012-11-09 09:28:46 【问题描述】:

我是 Java 开发人员,我曾经将我的 Java 实体作为 POJO 进行测试。现在,使用 Obj-C,我想对继承自 NSManagedObject 的实体执行相同的操作(我使用 CoreData 进行持久化)。

例如,我想测试我的客户实体:

-(void)myTest 
Customer *customer = [Customer alloc] init];
customer.name = @"toto";
GHAssertEqualStrings(customer.name, @"toto", @"");

但是我遇到的错误是:

NSInvalidArgumentException 原因:-[Customers setName:]:无法识别的选择器发送到实例...

所以我已经使用适当的数据库架构 url 在 setUp 中加载了所有 NSManagedObjectContext。现在我将我的客户实例化为它并且它可以工作:

Customers *customer = [NSEntityDescription
insertNewObjectForEntityForName:kDataBaseCustomerKey inManagedObjectContext:ctx];

但这是测试 'POJO' 的合适方法吗?我想在不加载任何模型的情况下测试我的 Customer 类,因为在这种情况下我不关心数据模型。

感谢您的建议。

问候。

【问题讨论】:

【参考方案1】:

这取决于您要测试的内容。从概念上将核心数据模型与现实生活中的实现分开有些困难。所以我通常在单元测试代码中构建自己的堆栈。另一方面,如果您真的在测试仅依赖于 NSManagedObject 子类实现的代码,那么我认为您概述的方法没有任何问题。

作为参考,如果您对如何为单元测试重新创建堆栈感兴趣,以下是我的做法。 (我刚刚注意到您使用的是GHUnit 而不是OCUnit - 它应该是相同的,但您可能需要确保该模型包含在GHUnit 应用程序的捆绑资源中。

创建一个继承自SenTestCase 的类,并使用NSInMemoryStoreType 在那里构建您的Core Data 堆栈:

@implementation CCFCoreDataTestCase 
    NSManagedObjectModel *_mom;
    NSPersistentStoreCoordinator *_psc;
    NSManagedObjectContext *_moc;
    NSPersistentStore *_store;


@synthesize managedObjectContext = _moc;

- (void)setUp 
    [super setUp];

    NSArray *bundles = [NSArray arrayWithObject:[NSBundle bundleForClass:[self class]]];
    _mom = [NSManagedObjectModel mergedModelFromBundles:bundles];
    _psc = [[NSPersistentStoreCoordinator alloc] initWithManagedObjectModel:_mom];

    _store = [_psc addPersistentStoreWithType:NSInMemoryStoreType configuration:nil URL:nil options:nil error:NULL];
    STAssertNotNil(_store,@"Unable to create in-memory store");

    _moc = [[NSManagedObjectContext alloc] initWithConcurrencyType:NSMainQueueConcurrencyType];
    [_moc setPersistentStoreCoordinator:_psc];


- (void)tearDown 
    [super tearDown];
    _mom = nil; _psc = nil; _moc = nil; _store = nil;

所有涉及核心数据的测试用例都应该继承自这个子类。假设您使用的是内存存储类型,并且每次测试都构建和拆除模型,您可以最小化任何依赖关系并从每个测试的未填充模型开始。当然,没有关于性能的承诺。

编辑:

我发现这篇文章在处理 Core Data 对象的单元测试时很有帮助 - Unit testing Core Data-driven apps

编辑 2:

Graham Lee(上述帖子的作者)有另一篇关于单元测试不依赖于模拟的 Core Data 对象的帖子。见this one我上面描述的模式和他在第二个链接中所做的比较一致。

【讨论】:

是的,我喜欢这种内存解决方案。谢谢它有效(即使使用 GHUnit)。但这意味着在我之前的解决方案中,我使用了包含现有数据的“生产”数据库? 供您的编辑:这篇文章很有趣,使用该解决方案我将能够在不依赖核心数据堆栈的情况下测试我的 MockCustomer。但是..我不喜欢为每个实体创建一个 Mock 类。这不是一个沉重的解决方案吗?你怎么看? 查看第二次编辑。我依赖于我描述的模式——在测试中创建一个轻量级堆栈。在您之前的解决方案中,至少对于 GHUnit,我认为您实际上并没有使用“生产”数据,这取决于您如何实例化 NSManagedObjectContext - 特别是因为 GHUnit 不知道您的应用程序的持久存储。 好的,谢谢您的回答。我也会依赖你提出的模型,但是 Lee 描述的 Mock 解决方案很有趣。

以上是关于NSManagedObject [GHUnit] 的单元测试的主要内容,如果未能解决你的问题,请参考以下文章

如何在命令行 GHUnit 中使用 UIApplicationDelegate?

GHUnit : 测试魔法记录代码

GHUnit 作为测试包而不是单独的目标

GHUnit,OCMock:如何测试两个指定块之一是不是被调用?

如何为委托方法编写 GHunit 测试用例?

GHUnit 跳转到代码中的错误