UIDocument openWithCompletionHandler: crash

2019-01-26 20:26发布

I'm using Core Data in my project and get a rare crash in the following code section

 -(void) useDocument{
     AFFormsCoreDataEngine* engine = [AFFormsCoreDataEngine sharedInstance];
     if (![[NSFileManager defaultManager] fileExistsAtPath: [engine.formsDatabase.fileURL path]])
     {
         [engine.formsDatabase saveToURL: engine.formsDatabase.fileURL forSaveOperation: UIDocumentSaveForCreating completionHandler: ^(BOOL success){
            // setup
         }];
    }
    else if (engine.formsDatabase.documentState == UIDocumentStateClosed)
    {
        [engine.formsDatabase openWithCompletionHandler: ^(BOOL success){
             // setup
        }];
    }
    else if (engine.formsDatabase.documentState == UIDocumentStateNormal)
    {
         // setup
    }
}

This is what crash log says:

Last Exception Backtrace:
0   CoreFoundation                  0x371fd88f __exceptionPreprocess + 163
1   libobjc.A.dylib                 0x31272259 objc_exception_throw + 33
2   CoreFoundation                  0x371fd789 +[NSException raise:format:] + 1
3   Foundation                      0x32ce83a3 -[NSAssertionHandler handleFailureInMethod:object:file:lineNumber:description:] + 91
4   UIKit                           0x306b3149 -[UIDocument openWithCompletionHandler:] + 173
5   EETECH                          0x00014d23 -[AFFormListViewController useDocument] (AFFormListViewController.m:150)

Can anyone help me to solve this issue? It happens very seldom, but still is very unpleasant

2条回答
干净又极端
2楼-- · 2019-01-26 20:52

Well, an assertion is failing in the UIDocument code. You should probably provide more code, because you are obviously setting something up improperly.

The assertion is then throwing an exception. If you @catch exceptions in this method, you can log the exception.

Or, you can assign your own NSAssertionHandler to the thread, and see the assertion directly.

查看更多
叛逆
3楼-- · 2019-01-26 21:00

The error occurs if your app attempts to call your useDocument method twice in close succession.

Because the openWithCompletionHandler: opens the document asynchronously, the document may still be opening when the method is called again.

If this happens, your app ends up trying to open the document twice (as the document state will remain UIDocumentStateClosed until completion) and this causes the exception to be thrown.

If you have an exception breakpoint, you may see something like this in the console:

 *** Terminating app due to uncaught exception 'NSInternalInconsistencyException', reason: 'attempt to open or a revert document that already has an open or revert operation in flight:
查看更多
登录 后发表回答