私はドメインモデルを使用してiphoneアプリケーションを開発しており、これまでアプリの永続性の側面を延期してきました。 Core Dataは、明確に定義されたモデルをすでに持っているので、本当に良いソリューションのように見えますが、既存の単体テストで問題が発生しています。
これが私が今持っているものの簡単な例です:
- (void)test_full_name_returns_correct_string {
Patient *patient = [[Patient alloc] init];
patient.firstName = @"charlie";
patient.lastName = @"chaplin";
STAssertTrue([[patient fullName] isEqualToString:@"charlie chaplin"], @"should have matched full name");
}
PatientオブジェクトがNSManagedObjectから拡張され、firstNameプロパティとlastNameプロパティに@dynamicを使用した後、これを機能させるにはどうすればよいですか?
他の誰かがCoreDataでこのタイプに遭遇しましたか?ありがとう。
各メソッド内または_-setUp
_内で、CoreDataスタックを構築してから破棄する必要があります。 NSInMemoryPersistentStore
を使用すると、単体テスト用に高速でメモリ内に保持されます。 TestCaseサブクラスに@property (nonatomic,retain) NSManagedObjectContext *moc
を追加します。次に:
_- (void)setUp {
NSManagedObjectModel *mom = [NSManagedObjectModel mergedModelFromBundles:[NSArray arrayWithObject:bundleContainingXCDataModel]];
NSPersistentStoreCoordinator *psc = [[NSPersistentStoreCoordinator alloc] initWithManagedObjectModel:mom];
STAssertTrue([psc addPersistentStoreWithType:NSInMemoryStoreType configuration:nil URL:nil options:nil error:NULL] ? YES : NO, @"Should be able to add in-memory store");
self.moc = [[NSManagedObjectContext alloc] init];
self.moc.persistentStoreCoordinator = psc;
[mom release];
[psc release];
}
- (void)tearDown {
self.moc = nil;
}
_
テスト方法は次のようになります。
_- (void)test_full_name_returns_correct_string {
Patient *patient = [NSEntityDescription insertNewObjectForEntityForName:@"Person" inManagedObjectContext:self.moc];
patient.firstName = @"charlie";
patient.lastName = @"chaplin";
STAssertTrue([[patient fullName] isEqualToString:@"charlie chaplin"], @"should have matched full name");
}
_
エンティティの名前がPerson
であると仮定します。ちなみに、お使いのバージョンのメソッドでメモリリークが発生しました。非コアデータバージョンでは、患者は_-release
_されている必要があります(_insertNewObjectForEntityForName:managedObjectContext:
_は自動解放されたインスタンスを返します)。
上記のBarryWarkによる回答を使用しましたが、現在のプロジェクトXcode 5、iOS 7で機能するように、いくつかの変更を加える必要がありました。
プロパティは同じままでした:
@interface SIDataTest : XCTestCase
@property (nonatomic, retain) NSManagedObjectContext *moc;
@end
セットアップは、最初にリリースしないように、次にモデルURLを提供するために、実際に変更する必要がありました。
- (void)setUp
{
[super setUp];
NSURL *modelURL = [[NSBundle mainBundle] URLForResource:@"SimpleInvoice" withExtension:@"momd"];
NSManagedObjectModel *mom = [[NSManagedObjectModel alloc] initWithContentsOfURL:modelURL];
NSPersistentStoreCoordinator *psc = [[NSPersistentStoreCoordinator alloc] initWithManagedObjectModel:mom];
XCTAssertTrue([psc addPersistentStoreWithType:NSInMemoryStoreType configuration:nil URL:nil options:nil error:NULL] ? YES : NO, @"Should be able to add in-memory store");
self.moc = [[NSManagedObjectContext alloc] init];
self.moc.persistentStoreCoordinator = psc;
}
テストケースの例を次に示します。
- (void)testCreateNew
{
Invoice *newInvoice = [NSEntityDescription insertNewObjectForEntityForName:@"Invoice" inManagedObjectContext:self.moc];
newInvoice.dueDate = [NSDate date];
NSString* title = [[NSString alloc] initWithFormat:@"Invoice %@", @112];
newInvoice.title = title;
// Save the context.
NSError *error = nil;
if (![self.moc save:&error]) {
// Replace this implementation with code to handle the error appropriately.
// abort() causes the application to generate a crash log and terminate. You should not use this function in a shipping application, although it may be useful during development.
XCTFail(@"Error saving in \"%s\" : %@, %@", __PRETTY_FUNCTION__, error, [error userInfo]);
}
XCTAssertFalse(self.moc.hasChanges,"All the changes should be saved");
}