尝试通过AVAssetWriter和AVAssetWriterInput方法将多个视频连接成一个视频时。在第3个视频之后,我从AVAssetWriter.error收到“无法编码”错误。
此外,我能够通过控制台看到成功读取到缓冲区,但只有最后成功读取的视频在连接的mov中结束。任何对任何一个或两个问题的了解,来源和日志如下。感谢
+(void)doWriteWithVideoWriter:(AVAssetWriter *)videoWriter withIndex:(int)index withWriterInputArray:(NSMutableArray *)writersArray withInstance:(VideoCombinerManager *)theInstance{
if ([writersArray count] > 0)
{
int newIndex = index+1;
NSError *readerError;
NSDictionary *videoOptions = [NSDictionary dictionaryWithObject:[NSNumber numberWithInt:kCVPixelFormatType_420YpCbCr8BiPlanarVideoRange] forKey:(id)kCVPixelBufferPixelFormatTypeKey];
AVAsset *sourceAsset = [theInstance.loadedAssetsForCompilationDictionary objectForKey:[theInstance.sortedKeysArray objectAtIndex:index]];
AVAssetTrack *videoTrack = [[sourceAsset tracksWithMediaType:AVMediaTypeVideo] objectAtIndex:0];
AVAssetReaderTrackOutput *currentReaderTrackOutput = [[AVAssetReaderTrackOutput alloc] initWithTrack:videoTrack outputSettings:videoOptions];
AVAssetReader *currentReader = [[AVAssetReader alloc] initWithAsset:sourceAsset error:&readerError];
[currentReader addOutput:currentReaderTrackOutput];
[currentReader startReading];
AVAssetWriterInput *writerInput = [[writersArray objectAtIndex:0] retain];
dispatch_queue_t _processingQueue = dispatch_queue_create("asdf", NULL);
[writerInput requestMediaDataWhenReadyOnQueue:_processingQueue usingBlock:^{
if ([writerInput isReadyForMoreMediaData])
{
CMSampleBufferRef nextSampleBuffer;
if ([currentReader status] == AVAssetReaderStatusReading &&
(nextSampleBuffer = [currentReaderTrackOutput copyNextSampleBuffer])) {
if (nextSampleBuffer)
{
BOOL result = [writerInput appendSampleBuffer:nextSampleBuffer];
if (!result)
{
NSLog(@"videoWriter.error.userInfo: %@", videoWriter.error.userInfo);
}
CFRelease(nextSampleBuffer);
}
}
else
{
NSLog(@"writer done: %d", index);
dispatch_release(_processingQueue);
[writersArray removeObjectAtIndex:0];
[writerInput markAsFinished];
[writerInput release];
[currentReader release];
[currentReaderTrackOutput release];
[VideoCombinerManager doWriteWithVideoWriter:videoWriter withIndex:newIndex withWriterInputArray:writersArray withInstance:theInstance];
}
}
}];
}
else
[videoWriter finishWriting];
}
作家完成:0
作家完成:1
作家完成:2
videoWriter.error.userInfo: {
NSLocalizedDescription = "Cannot Encode";
NSLocalizedFailureReason = "The encoder required for this media is busy.";
NSLocalizedRecoverySuggestion = "Stop any other actions that encode media and try again.";
NSUnderlyingError = "Error Domain=NSOSStatusErrorDomain Code=-12915 \"The operation couldn\U2019t be completed. (OSStatus error -12915.)\"";
}
videoWriter.error.userInfo: {
NSLocalizedDescription = "Cannot Encode";
NSLocalizedFailureReason = "The encoder required for this media is busy.";
NSLocalizedRecoverySuggestion = "Stop any other actions that encode media and try again.";
NSUnderlyingError = "Error Domain=NSOSStatusErrorDomain Code=-12915 \"The operation couldn\U2019t be completed. (OSStatus error -12915.)\"";
}
答案 0 :(得分:5)
对于Can not Encode错误,您同时存在太多的writer和/或writerInput。当我遇到同样的错误时,我意识到我必须释放所有以前使用的编写器和writerInputs来摆脱它。 既然您确实发布了它们,那么您的问题可能是您同时拥有其中的许多内容。