jjzjj

ios - AVAssetExportSession 间歇性错误 11820 "Cannot Complete Export"Suggestion=再次尝试导出

coder 2024-01-18 原文

EXPORT STATUS 4 Error Domain=AVFoundationErrorDomain Code=-11820 "Cannot Complete Export" UserInfo={NSLocalizedDescription=Cannot Complete Export, NSLocalizedRecoverySuggestion=Try exporting again.}

尝试导出包含 AVMutableVideoCompositionLayerInstructionAVMutableComposition 和使用 AVMutableVideoComposition 时遇到间歇性错误AVAssetExportSession.

目标是合并无限数量的视频并使用 layerInstructions 在剪辑之间应用过渡。

附言错误不一致。它在尝试合并 5 个剪辑和 18 个剪辑时有效,但在尝试合并 17 个剪辑时无效。

我已经在下面发布了我的代码。非常感谢任何帮助。

编辑:问题似乎与多个 AVMutableCompositionTrack(s) 的创建有关。如果创建超过 15 或 16 个,则会发生错误。但是,我认为创建多个 AVMutableCompositionTrack 是重叠所有视频和创建重叠过渡的必要条件。

编辑 2:选择较短的视频时,会在发生错误之前处理更多的视频。因此,它看起来像是一个内存问题,轨道正在被释放。但是,基于内存管理工具似乎没有内存泄漏。

-(void)prepareMutableCompositionForPlayback{
AVMutableComposition *mutableComposition = [[AVMutableComposition alloc] init];
AVMutableVideoCompositionInstruction *mainInstruction = [AVMutableVideoCompositionInstruction videoCompositionInstruction];
mainInstruction.backgroundColor = [[UIColor blackColor] CGColor];

NSMutableArray *instructionsArray = [[NSMutableArray alloc] init];

videoStartTime = kCMTimeZero;

for(int i = 0; i < videoAssetsArray.count; i++){
    AVAsset *videoAsset = [videoAssetsArray objectAtIndex:i];
    CMTime currentVideoDuration = [videoAsset duration];

    AVMutableCompositionTrack *videoTrack = [mutableComposition addMutableTrackWithMediaType:AVMediaTypeVideo preferredTrackID:kCMPersistentTrackID_Invalid];
    [videoTrack insertTimeRange:CMTimeRangeMake(kCMTimeZero, currentVideoDuration) ofTrack:[[videoAsset tracksWithMediaType:AVMediaTypeVideo] objectAtIndex:0] atTime:videoStartTime error:nil];

    CGSize videoSize = [videoTrack naturalSize];

    if([videoAsset tracksWithMediaType:AVMediaTypeAudio].count > 0){
        AVMutableCompositionTrack *audioTrack = [mutableComposition addMutableTrackWithMediaType:AVMediaTypeAudio preferredTrackID:kCMPersistentTrackID_Invalid];
        [audioTrack insertTimeRange:CMTimeRangeMake(kCMTimeZero, currentVideoDuration) ofTrack:[[videoAsset tracksWithMediaType:AVMediaTypeAudio] objectAtIndex:0] atTime:videoStartTime error:nil];
    }

    //INSTRUCTIONS - TRANSITIONS
    AVMutableVideoCompositionLayerInstruction *layerInstruction = [AVMutableVideoCompositionLayerInstruction videoCompositionLayerInstructionWithAssetTrack:videoTrack];

    int transitionNumber = [[videoTransitionsArray objectAtIndex:i] intValue];
    float transitionDuration = [[videoTransitionsDurationArray objectAtIndex:i] floatValue];

    if(i == 0){
        [layerInstruction setOpacityRampFromStartOpacity:1.0 toEndOpacity:0.0 timeRange:CMTimeRangeMake(CMTimeSubtract(currentVideoDuration, CMTimeMakeWithSeconds(transitionDuration, 600)), CMTimeMakeWithSeconds(transitionDuration, 600))];
    }
    else{
        int previousTransitionNumber = [[videoTransitionsArray objectAtIndex:i - 1] intValue];
        float previousTransitionDuration = [[videoTransitionsDurationArray objectAtIndex:i - 1] floatValue];

        if(i < videoAssetsArray.count - 1){
            [layerInstruction setOpacityRampFromStartOpacity:1.0 toEndOpacity:1.0 timeRange:CMTimeRangeMake(videoStartTime, CMTimeMakeWithSeconds(previousTransitionDuration, 600))];

            [layerInstruction setOpacityRampFromStartOpacity:1.0 toEndOpacity:0.0 timeRange:CMTimeRangeMake(CMTimeAdd(videoStartTime, CMTimeSubtract(currentVideoDuration, CMTimeMakeWithSeconds(transitionDuration, 600))), CMTimeMakeWithSeconds(transitionDuration, 600))];
        }
        else{
            [layerInstruction setOpacityRampFromStartOpacity:1.0 toEndOpacity:1.0 timeRange:CMTimeRangeMake(videoStartTime, CMTimeMakeWithSeconds(previousTransitionDuration, 600))];
        }
    }

    [instructionsArray addObject:layerInstruction];

    if(i < videoAssetsArray.count - 1){
        //TAKING INTO ACCOUNT THE TRANSITION DURATION TO OVERLAP VIDEOS
        videoStartTime = CMTimeAdd(videoStartTime, CMTimeSubtract(currentVideoDuration, CMTimeMakeWithSeconds(transitionDuration, 600)));
    }
    else{
        //TRANSITION NOT APPLIED TO THE END OF THE LAST CLIP
        videoStartTime = CMTimeAdd(videoStartTime, currentVideoDuration);
    }
}

mainInstruction.timeRange = CMTimeRangeMake(kCMTimeZero,videoStartTime);
mainInstruction.layerInstructions = instructionsArray;

AVMutableVideoComposition *videoComposition = [AVMutableVideoComposition videoComposition];
videoComposition.instructions = [NSArray arrayWithObjects:mainInstruction,nil];
videoComposition.frameDuration = CMTimeMake(1, 30);
videoComposition.renderSize = CGSizeMake(1920, 1080);

NSArray* paths = NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES);
NSString *documentsDirectory = [paths objectAtIndex:0];
NSString *videoOutputPath = [documentsDirectory stringByAppendingPathComponent:@"videoRecordingFinalOutput.mov"];
NSURL *videoOutputURL = [[NSURL alloc] initFileURLWithPath:videoOutputPath];

AVAssetExportSession *videoExportSession = [[AVAssetExportSession alloc] initWithAsset:mutableComposition presetName:AVAssetExportPresetHighestQuality];
videoExportSession.outputURL = videoOutputURL;
videoExportSession.videoComposition = videoComposition;
videoExportSession.outputFileType = AVFileTypeQuickTimeMovie;

[videoExportSession exportAsynchronouslyWithCompletionHandler:^{
    NSLog(@"EXPORT STATUS %ld %@", (long)videoExportSession.status, videoExportSession.error);

    if(videoExportSession.error == NULL){
        NSLog(@"EXPORT SUCCESSFUL");

        [library writeVideoAtPathToSavedPhotosAlbum:videoOutputURL
                                    completionBlock:^(NSURL *assetURL, NSError *error) {
                                        if(error) {

                                            NSError *error = nil;
                                            if([[NSFileManager defaultManager] fileExistsAtPath:videoOutputPath]){
                                                [[NSFileManager defaultManager] removeItemAtPath:videoOutputPath error:&error];
                                                if(error){
                                                    NSLog(@"VIDEO FILE DELETE FAILED");
                                                }
                                                else{
                                                    NSLog(@"VIDEO FILE DELETED");
                                                }
                                            }
                                        }
                                        else{
                                            NSError *error = nil;
                                            if([[NSFileManager defaultManager] fileExistsAtPath:videoOutputPath]){
                                                [[NSFileManager defaultManager] removeItemAtPath:videoOutputPath error:&error];
                                                if(error){
                                                    NSLog(@"VIDEO FILE DELETE FAILED");
                                                }
                                                else{
                                                    NSLog(@"VIDEO FILE DELETED");
                                                }
                                            }
                                        }
                                    }];
    }
    else{
        NSError *error = nil;
        if([[NSFileManager defaultManager] fileExistsAtPath:videoOutputPath]){
            [[NSFileManager defaultManager] removeItemAtPath:videoOutputPath error:&error];
            if(error){
                NSLog(@"VIDEO FILE DELETE FAILED");
            }
            else{
                NSLog(@"VIDEO FILE DELETED");
            }
        }
    }
}];
}

最佳答案

与其为每个剪辑创建新的 videoTracks,不如尝试仅使用 2 个 videoTracks 并在这 2 个中插入 timeRanges。并在 2 个轨道之间进行过渡

所以第一个视频将被插入到 videoTrack1 中,第二个视频将被插入到 videoTrack2 中,这样就可以应用过渡,然后再次将第三个剪辑插入到轨道 1 中,依此类推。

关于ios - AVAssetExportSession 间歇性错误 11820 "Cannot Complete Export"Suggestion=再次尝试导出,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/35170799/

有关ios - AVAssetExportSession 间歇性错误 11820 "Cannot Complete Export"Suggestion=再次尝试导出的更多相关文章

  1. ruby-on-rails - rails : "missing partial" when calling 'render' in RSpec test - 2

    我正在尝试测试是否存在表单。我是Rails新手。我的new.html.erb_spec.rb文件的内容是:require'spec_helper'describe"messages/new.html.erb"doit"shouldrendertheform"dorender'/messages/new.html.erb'reponse.shouldhave_form_putting_to(@message)with_submit_buttonendendView本身,new.html.erb,有代码:当我运行rspec时,它失败了:1)messages/new.html.erbshou

  2. ruby-on-rails - 由于 "wkhtmltopdf",PDFKIT 显然无法正常工作 - 2

    我在从html页面生成PDF时遇到问题。我正在使用PDFkit。在安装它的过程中,我注意到我需要wkhtmltopdf。所以我也安装了它。我做了PDFkit的文档所说的一切......现在我在尝试加载PDF时遇到了这个错误。这里是错误:commandfailed:"/usr/local/bin/wkhtmltopdf""--margin-right""0.75in""--page-size""Letter""--margin-top""0.75in""--margin-bottom""0.75in""--encoding""UTF-8""--margin-left""0.75in""-

  3. ruby - 检查 "command"的输出应该包含 NilClass 的意外崩溃 - 2

    为了将Cucumber用于命令行脚本,我按照提供的说明安装了arubagem。它在我的Gemfile中,我可以验证是否安装了正确的版本并且我已经包含了require'aruba/cucumber'在'features/env.rb'中为了确保它能正常工作,我写了以下场景:@announceScenario:Testingcucumber/arubaGivenablankslateThentheoutputfrom"ls-la"shouldcontain"drw"假设事情应该失败。它确实失败了,但失败的原因是错误的:@announceScenario:Testingcucumber/ar

  4. ruby-on-rails - 迷你测试错误 : "NameError: uninitialized constant" - 2

    我遵循MichaelHartl的“RubyonRails教程:学习Web开发”,并创建了检查用户名和电子邮件长度有效性的测试(名称最多50个字符,电子邮件最多255个字符)。test/helpers/application_helper_test.rb的内容是:require'test_helper'classApplicationHelperTest在运行bundleexecraketest时,所有测试都通过了,但我看到以下消息在最后被标记为错误:ERROR["test_full_title_helper",ApplicationHelperTest,1.820016791]test

  5. ruby-on-rails - 相关表上的范围为 "WHERE ... LIKE" - 2

    我正在尝试从Postgresql表(table1)中获取数据,该表由另一个相关表(property)的字段(table2)过滤。在纯SQL中,我会这样编写查询:SELECT*FROMtable1JOINtable2USING(table2_id)WHEREtable2.propertyLIKE'query%'这工作正常:scope:my_scope,->(query){includes(:table2).where("table2.property":query)}但我真正需要的是使用LIKE运算符进行过滤,而不是严格相等。然而,这是行不通的:scope:my_scope,->(que

  6. 使用 ACL 调用 upload_file 时出现 Ruby S3 "Access Denied"错误 - 2

    我正在尝试编写一个将文件上传到AWS并公开该文件的Ruby脚本。我做了以下事情:s3=Aws::S3::Resource.new(credentials:Aws::Credentials.new(KEY,SECRET),region:'us-west-2')obj=s3.bucket('stg-db').object('key')obj.upload_file(filename)这似乎工作正常,除了该文件不是公开可用的,而且我无法获得它的公共(public)URL。但是当我登录到S3时,我可以正常查看我的文件。为了使其公开可用,我将最后一行更改为obj.upload_file(file

  7. ruby - 如何验证 IO.copy_stream 是否成功 - 2

    这里有一个很好的答案解释了如何在Ruby中下载文件而不将其加载到内存中:https://stackoverflow.com/a/29743394/4852737require'open-uri'download=open('http://example.com/image.png')IO.copy_stream(download,'~/image.png')我如何验证下载文件的IO.copy_stream调用是否真的成功——这意味着下载的文件与我打算下载的文件完全相同,而不是下载一半的损坏文件?documentation说IO.copy_stream返回它复制的字节数,但是当我还没有下

  8. ruby - 安装 Ruby 时遇到问题(无法下载资源 "readline--patch") - 2

    当我尝试安装Ruby时遇到此错误。我试过查看this和this但无济于事➜~brewinstallrubyWarning:YouareusingOSX10.12.Wedonotprovidesupportforthispre-releaseversion.Youmayencounterbuildfailuresorotherbreakages.Pleasecreatepull-requestsinsteadoffilingissues.==>Installingdependenciesforruby:readline,libyaml,makedepend==>Installingrub

  9. Ruby 文件 IO 定界符? - 2

    我正在尝试解析一个文本文件,该文件每行包含可变数量的单词和数字,如下所示:foo4.500bar3.001.33foobar如何读取由空格而不是换行符分隔的文件?有什么方法可以设置File("file.txt").foreach方法以使用空格而不是换行符作为分隔符? 最佳答案 接受的答案将slurp文件,这可能是大文本文件的问题。更好的解决方案是IO.foreach.它是惯用的,将按字符流式传输文件:File.foreach(filename,""){|string|putsstring}包含“thisisanexample”结果的

  10. ruby - RVM "ERROR: Unable to checkout branch ."单用户 - 2

    我在新的Debian6VirtualBoxVM上安装RVM时遇到问题。我已经安装了所有需要的包并使用下载了安装脚本(curl-shttps://rvm.beginrescueend.com/install/rvm)>rvm,但以单个用户身份运行时bashrvm我收到以下错误消息:ERROR:Unabletocheckoutbranch.安装在这里停止,并且(据我所知)没有安装RVM的任何文件。如果我以root身份运行脚本(对于多用户安装),我会收到另一条消息:Successfullycheckedoutbranch''安装程序继续并指示成功,但未添加.rvm目录,甚至在修改我的.bas

随机推荐