swift - 单元测试CoreData(+ MagicalRecord)模型触发EXC_BAD_ACCESS

Yam*_*man 9 unit-testing core-data ios magicalrecord swift

我需要XCTest单独测试()我的一些方法,包括对CoreData模型的引用.

以下行正确执行:

var airport: AnyObject! = Airport.MR_createEntity()

(lldb) po airport <Airport: 0x7fcf54216940> (entity: Airport; id: 0x7fcf54216a20 <x-coredata:///Airport/t1D3D08DA-70F9-4DA0-9487-BD6047EE93692> ; data: {
    open = nil;
    shortName = nil;
    visible = nil; })
Run Code Online (Sandbox Code Playgroud)

而以下行触发EXC_BAD_ACCESS:

var airport2: Airport = Airport.MR_createEntity() as! Airport

(lldb) po airport2
error: Execution was interrupted, reason: EXC_BAD_ACCESS (code=1, address=0x0).
The process has been returned to the state before expression evaluation.
Run Code Online (Sandbox Code Playgroud)

我的主要目标没有出现此错误的迹象.配置是:两个目标中的模型对象,前缀为@objc(MyModel)类,在我的类中的模型中没有命名空间xcdatamodel

知道这里发生了什么吗?

Dan*_*sko 3

是的,所以我终于弄清楚了这件事的真相,但它并不漂亮。实际上有一个针对此问题的雷达,因为它似乎是 Swift 编译器无法识别测试目标中的 ManagedObject 转换的错误。所以把你的声音加入到噪音中

从定义如下的实体开始:

@objc(Member)
class Member: NSManagedObject {    
    @NSManaged var name: String
}
Run Code Online (Sandbox Code Playgroud)

我编写了一个简单的测试类,在其中以 3 种不同的方式创建 MO:

前两个失败了:

let context = NSManagedObjectContext.MR_defaultContext()

func testMagicalRecordCreation() {
    let m = Member.MR_createInContext(context) as? Member
    XCTAssertNotNil(m, "Failed to create object")//fails
}

func testEntityDescriptionClassCreation() {
    let m2 = NSEntityDescription.insertNewObjectForEntityForName("Member", inManagedObjectContext: context) as? Member
    XCTAssertNotNil(m2, "Failed to create object")//fails
}
Run Code Online (Sandbox Code Playgroud)

然后就成功了

func testManualMOCreation() {
    let ent = NSEntityDescription.entityForName("Member", inManagedObjectContext: context)!
    let m3 = Member(entity: ent, insertIntoManagedObjectContext: context)
    XCTAssertNotNil(m3, "Failed to create object")
}
Run Code Online (Sandbox Code Playgroud)

这意味着现在您有两个选择。用 Objective-C 编写测试;或者创建一个实用方法,使用我上面展示的方法将测试对象插入到上下文中。

这里有一篇关于这种行为的好文章

我最终使用了 NSManagedObjectContext 扩展来在 Swift 测试中显式使用:

extension NSManagedObjectContext {
    func insertTestEntity<T: NSManagedObject>(entity: T.Type) -> T {
        let entityDescription = NSEntityDescription.entityForName(NSStringFromClass(T.self), inManagedObjectContext: self)!
        return T(entity: entityDescription, insertIntoManagedObjectContext: self)
    }
}
Run Code Online (Sandbox Code Playgroud)

它可以这样使用:

func testConvenienceCreation() {
    let m4 = context.insertTestEntity(Member)
    XCTAssertNotNil(m4, "Failed to create object")
}
Run Code Online (Sandbox Code Playgroud)

更多关于这种方法的阅读请点击这里