我的应用程序是一个自定义格式的查看器,这是一个具有明确定义的 XML list 和资源(例如图像和电影)的 zip 文件。我使用 zlib 打开内存中的 zip 文件,然后继续显示所述资源。
我遇到的一个问题是我无法正确显示视频,显然是因为 QTMovie 无法确定 mime 类型。从文件([QTMovie movieWithFile])加载的电影效果很好。从内存加载([QTMovie movieWithData])拒绝工作。
这是有道理的,因为缺少文件扩展名,QTMovie 无法确定 mime 类型信息。经过一番搜索后,我采用以下方式使用 QTDataReference:
NSData *movieData = ...read from memory...;
QTDataReference *movieDataReference = [[QTDataReference alloc] initWithReferenceToData:movieData name:fileName MIMEType:@"video/x-m4v"];
QTMovie *mov = [QTMovie movieWithDataReference:movieDataReference error:&err];
这工作得很好,但是硬编码 MIMEType 远非理想。我可以访问文件名和扩展名,因此我尝试使用 UTI 查找 mime 类型(感谢 #macdev 上的好心人):
- (NSString*)mimeTypeForExtension:(NSString*)ext {
CFStringRef UTI = UTTypeCreatePreferredIdentifierForTag(kUTTagClassFilenameExtension,(CFStringRef)ext,NULL);
return NSMakeCollectable(UTTypeCopyPreferredTagWithClass((CFStringRef)UTI,kUTTagClassMIMEType));
}
但这不起作用。显然,某个地方有一个内部 OS X 扩展数据库和相应的 mime 类型。否则磁盘上的电影将无法工作。我如何访问它?
谢谢!
最佳答案
您遇到的问题是 m4v 和 m4p 没有向 Launch Services 注册 mime 类型(可能是因为 m4v 和 m4p 的 mime 类型不是标准的)。无论如何,您可能应该做的是处理这样的边缘情况,然后在函数返回时检查 nil(以防扩展未注册且未由您处理)。
另一件事是您当前的使用正在泄漏内存。我假设您正在使用垃圾收集,但第一次调用创建了一个从未释放的 CFString。以下是您的方法的建议实现:
-(NSString*)mimeTypeForExtension:(NSString*)ext
{
NSAssert( ext, @"Extension cannot be nil" );
NSString* mimeType = nil;
CFStringRef UTI = UTTypeCreatePreferredIdentifierForTag(kUTTagClassFilenameExtension,
(CFStringRef)ext, NULL);
if( !UTI ) return nil;
CFStringRef registeredType = UTTypeCopyPreferredTagWithClass(UTI, kUTTagClassMIMEType);
if( !registeredType ) // check for edge case
{
if( [ext isEqualToString:@"m4v"] )
mimeType = @"video/x-m4v";
else if( [ext isEqualToString:@"m4p"] )
mimeType = @"audio/x-m4p";
// handle anything else here that you know is not registered
} else {
mimeType = NSMakeCollectable(registeredType);
}
CFRelease(UTI);
return mimeType;
}
关于cocoa - 使用 Cocoa (OS X) 进行内存中 mime 类型检测?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/300618/