在我的应用程序中,调用[AVURLAsset isPlayableExtendedMIMEType:@"video/mp4; codecs=\"avc1.64001F, mp4a.40.2\""]
返回YES
,这是预期的。
如果我在单元测试中运行完全相同的代码,则它将返回NO
。
应用程序和单元测试均在运行iOS 7.0的iPhone Retina(4英寸)模拟器上使用Xcode 5.0.2运行。
- (void) testPlayableExtendedMIMEType
{
XCTAssertTrue([AVURLAsset class], @"");
XCTAssertTrue([AVURLAsset isPlayableExtendedMIMEType:@"video/mp4; codecs=\"avc1.64001F, mp4a.40.2\""], @"");
}
第一个断言通过,但是第二个断言失败。
为什么这种行为在单元测试和应用程序中会有所不同?
最佳答案
经过反向工程如何实现+[AVURLAsset isPlayableExtendedMIMEType:]
后,我找到了问题的原因。
这是它在做什么的堆栈跟踪:
frame #0: 0x01b2861e CoreMedia`CelestialGetModelSpecificName
frame #1: 0x01b2885a CoreMedia`CelestialCFCreatePropertyListFromBundleIdentifier + 11
frame #2: 0x00050039 AVFoundation`__33+[AVURLAsset _avfValidationPlist]_block_invoke_0 + 39
frame #3: 0x02e99014 libdispatch.dylib`_dispatch_client_callout + 14
frame #4: 0x02e8b09f libdispatch.dylib`dispatch_once_f + 57
frame #5: 0x02e8b061 libdispatch.dylib`dispatch_once + 31
frame #6: 0x00050006 AVFoundation`+[AVURLAsset _avfValidationPlist] + 49
frame #7: 0x00050664 AVFoundation`+[AVURLAsset isPlayableExtendedMIMEType:] + 64
CelestialCFCreatePropertyListFromBundleIdentifier
函数尝试读取MediaValidator.plist
框架内的MediaToolbox
文件。 plist文件的位置取决于设备型号名称。这是iOS 7模拟器的不同MediaValidator.plist
文件。MediaToolbox.framework
|-- J1
| `-- MediaValidator.plist
|-- K93
| `-- MediaValidator.plist
|-- N41
| `-- MediaValidator.plist
`-- N94
`-- MediaValidator.plist
运行应用程序时,
CelestialGetModelSpecificName
函数返回N41
,i.e. iPhone 5。运行单元测试时,
CelestialGetModelSpecificName
函数返回N88
,即iPhone 3GS。如您所见,MediaToolbox框架内没有N88
目录,这就是+[AVURLAsset isPlayableExtendedMIMEType:]
最终失败的原因。仔细研究
CelestialGetModelSpecificName
函数可以找到解决方案。模拟器读取IPHONE_SIMULATOR_CLASS
环境变量以了解正在模拟的设备。如果未设置IPHONE_SIMULATOR_CLASS
环境变量,则默认为硬编码的N88
值。因此,为了使测试通过,我们只需要手动将
IPHONE_SIMULATOR_CLASS
环境变量设置为N41
,因为单元测试运行程序不会自动设置它。setenv("IPHONE_SIMULATOR_CLASS", "N41", 0);